Come intercettare su UIViewAlertForUnsatisfiableConstraints?


234

Viene visualizzato un errore nel registro del debugger:

Will attempt to recover by breaking constraint 
<NSLayoutConstraint:0x191f0920 H:[MPKnockoutButton:0x17a876b0]-(34)-[MPDetailSlider:0x17a8bc50](LTR)>

Make a symbolic breakpoint at UIViewAlertForUnsatisfiableConstraints to catch this in the debugger.
The methods in the UIConstraintBasedLayoutDebugging category on UIView listed in <UIKit/UIView.h> may also be helpful.

Come posso intercettare questa chiamata? Non appare da nessuna parte nel mio codice.

screenshot1


In 9 casi su 10: questo è causato solo da: per qualche vista o oggetto sullo storyboard yoru, deseleziona "Installato". (Ad esempio, solo un pulsante di sviluppo o qualcosa di cui non hai più bisogno.) In generale, gestisce male "non installato": spesso lascia vincoli che diventano insignificanti senza l'elemento non installato. Spesso la soluzione è semplicemente quella di eliminare gli oggetti che hai dimenticato, che si trovano intorno a "non installati" - basta eliminarli.
Fattie

Risposte:


442

Questo post mi ha aiutato MOLTO !

Ho aggiunto UIViewAlertForUnsatisfiableConstraints breakpoint simbolico con l'azione suggerita:

Progetto Obj-C

po [[UIWindow keyWindow] _autolayoutTrace]

Breakpoint simbolico con azione personalizzata nel progetto Objective-C

Progetto rapido

expr -l objc++ -O -- [[UIWindow keyWindow] _autolayoutTrace]

Breakpoint simbolico con azione personalizzata

Con questo suggerimento, il registro è diventato più dettagliato ed è stato più facile per me identificare quale vista aveva il vincolo rotto.

UIWindow:0x7f88a8e4a4a0
|   UILayoutContainerView:0x7f88a8f23b70
|   |   UINavigationTransitionView:0x7f88a8ca1970
|   |   |   UIViewControllerWrapperView:0x7f88a8f2aab0
|   |   |   |   UIView:0x7f88a8ca2880
|   |   |   |   |   *UIView:0x7f88a8ca2a10
|   |   |   |   |   |   *UIButton:0x7f88a8c98820'Archived'
|   |   |   |   |   |   |   UIButtonLabel:0x7f88a8cb0e30'Archived'
|   |   |   |   |   |   *UIButton:0x7f88a8ca22d0'Download'
|   |   |   |   |   |   |   UIButtonLabel:0x7f88a8cb04e0'Download'
|   |   |   |   |   |   *UIButton:0x7f88a8ca1580'Deleted'
|   |   |   |   |   |   |   UIButtonLabel:0x7f88a8caf100'Deleted'
|   |   |   |   |   *UIView:0x7f88a8ca33e0
|   |   |   |   |   *_UILayoutGuide:0x7f88a8ca35b0
|   |   |   |   |   *_UILayoutGuide:0x7f88a8ca4090
|   |   |   |   |   _UIPageViewControllerContentView:0x7f88a8f1a390
|   |   |   |   |   |   _UIQueuingScrollView:0x7f88aa031c00
|   |   |   |   |   |   |   UIView:0x7f88a8f38070
|   |   |   |   |   |   |   UIView:0x7f88a8f381e0
|   |   |   |   |   |   |   |   UIView:0x7f88a8f39fa0, MISSING HOST CONSTRAINTS
|   |   |   |   |   |   |   |   |   *UIButton:0x7f88a8cb9bf0'Retrieve data'- AMBIGUOUS LAYOUT for UIButton:0x7f88a8cb9bf0'Retrieve data'.minX{id: 170}, UIButton:0x7f88a8cb9bf0'Retrieve data'.minY{id: 171}
|   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8f3ad80- AMBIGUOUS LAYOUT for UIImageView:0x7f88a8f3ad80.minX{id: 172}, UIImageView:0x7f88a8f3ad80.minY{id: 173}
|   |   |   |   |   |   |   |   |   *App.RecordInfoView:0x7f88a8cbe530- AMBIGUOUS LAYOUT for App.RecordInfoView:0x7f88a8cbe530.minX{id: 174}, App.RecordInfoView:0x7f88a8cbe530.minY{id: 175}, App.RecordInfoView:0x7f88a8cbe530.Width{id: 176}, App.RecordInfoView:0x7f88a8cbe530.Height{id: 177}
|   |   |   |   |   |   |   |   |   |   +UIView:0x7f88a8cc1d30- AMBIGUOUS LAYOUT for UIView:0x7f88a8cc1d30.minX{id: 178}, UIView:0x7f88a8cc1d30.minY{id: 179}, UIView:0x7f88a8cc1d30.Width{id: 180}, UIView:0x7f88a8cc1d30.Height{id: 181}
|   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8cc1ec0- AMBIGUOUS LAYOUT for UIView:0x7f88a8cc1ec0.minX{id: 153}, UIView:0x7f88a8cc1ec0.minY{id: 151}, UIView:0x7f88a8cc1ec0.Width{id: 154}, UIView:0x7f88a8cc1ec0.Height{id: 165}
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8e68e10- AMBIGUOUS LAYOUT for UIView:0x7f88a8e68e10.minX{id: 155}, UIView:0x7f88a8e68e10.minY{id: 150}, UIView:0x7f88a8e68e10.Width{id: 156}
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e65de0- AMBIGUOUS LAYOUT for UIImageView:0x7f88a8e65de0.minX{id: 159}, UIImageView:0x7f88a8e65de0.minY{id: 182}
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8e69080'8-6-2015'- AMBIGUOUS LAYOUT for UILabel:0x7f88a8e69080'8-6-2015'.minX{id: 183}, UILabel:0x7f88a8e69080'8-6-2015'.minY{id: 184}, UILabel:0x7f88a8e69080'8-6-2015'.Width{id: 185}
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cc0690'16:34'- AMBIGUOUS LAYOUT for UILabel:0x7f88a8cc0690'16:34'.minX{id: 186}, UILabel:0x7f88a8cc0690'16:34'.minY{id: 187}, UILabel:0x7f88a8cc0690'16:34'.Width{id: 188}, UILabel:0x7f88a8cc0690'16:34'.Height{id: 189}
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8cc2050- AMBIGUOUS LAYOUT for UIView:0x7f88a8cc2050.minX{id: 161}, UIView:0x7f88a8cc2050.minY{id: 166}, UIView:0x7f88a8cc2050.Width{id: 163}
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e69d90- AMBIGUOUS LAYOUT for UIImageView:0x7f88a8e69d90.minX{id: 190}, UIImageView:0x7f88a8e69d90.minY{id: 191}, UIImageView:0x7f88a8e69d90.Width{id: 192}, UIImageView:0x7f88a8e69d90.Height{id: 193}
|   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3cc00
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8e618d0
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e5ba10
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3cd70
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e58e10
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e5e7a0
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3cee0
|   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3dc70
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8e64dd0
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8e65290'Average flow rate'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8e712d0'177.0 ml/s'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8c97150'1299.4'
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3dde0
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8f3df50'Maximum flow rate'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cbfdb0'371.6 ml/s'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cc0230'873.5'
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3e2a0
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8f3e410'Total volume'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cc0f20'371.6 ml'
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3e870
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8f3ea00'Time do max. flow'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cc0ac0'3.6 s'
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3ee10
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8f3efa0'Flow time'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cbf980'2.1 s'
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3f3e0
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8f3f570'Voiding time'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cc17e0'3.5 s'
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3f9a0
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8f3fb30'Voiding delay'
|   |   |   |   |   |   |   |   |   |   |   |   *UILabel:0x7f88a8cc1380'1.0 s'
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8e65000
|   |   |   |   |   |   |   |   |   |   |   |   *UIButton:0x7f88a8e52f20'Show'
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e6e1d0
|   |   |   |   |   |   |   |   |   |   |   |   *UIButton:0x7f88a8e52c90'Send'
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e61bb0
|   |   |   |   |   |   |   |   |   |   |   |   *UIButton:0x7f88a8e528e0'Delete'
|   |   |   |   |   |   |   |   |   |   |   |   *UIImageView:0x7f88a8e6b3f0
|   |   |   |   |   |   |   |   |   |   |   |   *UIView:0x7f88a8f3ff60
|   |   |   |   |   |   |   |   |   *UIActivityIndicatorView:0x7f88a8cba080
|   |   |   |   |   |   |   |   |   |   UIImageView:0x7f88a8cba700
|   |   |   |   |   |   |   |   |   *_UILayoutGuide:0x7f88a8cc3150
|   |   |   |   |   |   |   |   |   *_UILayoutGuide:0x7f88a8cc3b10
|   |   |   |   |   |   |   UIView:0x7f88a8f339c0
|   |   UINavigationBar:0x7f88a8c96810
|   |   |   _UINavigationBarBackground:0x7f88a8e45c00
|   |   |   |   UIImageView:0x7f88a8e46410
|   |   |   UINavigationItemView:0x7f88a8c97520'App'
|   |   |   |   UILabel:0x7f88a8c97cc0'App'
|   |   |   UINavigationButton:0x7f88a8e3e850
|   |   |   |   UIImageView:0x7f88a8e445b0
|   |   |   _UINavigationBarBackIndicatorView:0x7f88a8f2b530

Legend:
    * - is laid out with auto layout
    + - is laid out manually, but is represented in the layout engine because translatesAutoresizingMaskIntoConstraints = YES
     - layout engine host

Quindi ho messo in pausa l'esecuzione Pausa e ho cambiato il colore di sfondo della vista problematica con il comando (sostituendo ovviamente 0x7f88a8cc2050con l'indirizzo di memoria del tuo oggetto ) ...

Obj-C

expr ((UIView *)0x7f88a8cc2050).backgroundColor = [UIColor redColor]

Swift 3.0

expr -l Swift -- import UIKit
expr -l Swift -- unsafeBitCast(0x7f88a8cc2050, to: UIView.self).backgroundColor = UIColor.red

... e il risultato è stato fantastico!

Vista suggerita

Semplicemente stupefacente! Spero che sia d'aiuto.


3
@iAnurag È possibile eseguire comandi nell'area della console quando l'esecuzione è in pausa.
Thomás Calmon,

2
@ TomCalmon Ho fatto lo stesso ... ma mostra il seguente errore rror: Execution was interrupted, reason: EXC_BAD_ACCESS (code=1, address=0x7f88a8cc2050). The process has been returned to the state before expression evaluation.
iAnurag il

2
expr -l objc++ -O -- [[UIWindow keyWindow] _autolayoutTrace]ritorna nilper me
Igor Andreev

2
Assicurarsi di sostituire 0x7f88a8cc2050 con l'indirizzo di memoria dell'oggetto ed eseguire il comando nella console quando l'esecuzione viene messa in pausa.
Tom Howard,

3
Unfreakingbelievable. Ottimo consiglio qui, mi ha aiutato totalmente ad andare direttamente al problema. Una volta che l'elemento è diventato rosso, continua l'esecuzione se possibile e vedrai l'evidenziazione.
Aaron,

255

Ti consigliamo di aggiungere un Symbolic Breakpoint. Apple fornisce un'eccellente guida su come farlo.

  1. Apri Breakpoint Navigator cmd+7( cmd+8in Xcode 9)
  2. Fai clic sul Addpulsante in basso a sinistra
  3. Selezionare Add Symbolic Breakpoint...
  4. Dove dice Symbolbasta digitareUIViewAlertForUnsatisfiableConstraints

Puoi anche trattarlo come qualsiasi altro punto di interruzione, accenderlo e spegnerlo, aggiungere azioni o registrare messaggi.


55
Semplicemente non capisco come posso eseguire il debug del problema meglio con questo suggerimento. Ho aggiunto un punto di interruzione simbolico ma non mi dà ancora abbastanza informazioni sul problema. L'unico modo è cercare di leggere riga per riga e capire quali sono le cause del problema .... altrimenti cancellare i vincoli e aggiungerli di nuovo insieme con l'anteprima nella vista asisstent dovrebbe aiutare di più!
Alex Cio,

11
Questo potrebbe aiutare a ottenere maggiori informazioni dopo essersi fermati al punto di interruzione: staxmanade.com/2015/06/debugging-ios-autolayout-issues
fabb

1
Aggiungo solo che ora puoi assegnare identificatori ai vincoli direttamente in IB, quindi quando li esegui il debug, questo è il nome che vedrai.
Mark A. Donohoe,

2
(follow-up su @MarqueIV) NSLayoutConstraintha una identifierproprietà da iOS 7 - Xcode 7 e versioni successive , che può essere impostata sia dagli storyboard IB sia dal codice. Impostando l'identificatore è possibile distinguere più facilmente tra i vincoli generati dal sistema e generati dall'utente nel registro di debug, ad esempio myConstraint.identifier = "centered image"(fonte ed esempi: useyourloaf.com/blog/using-identifiers-to-debug-autolayout )
PDK

@AlexCio In che modo aiuta? Il minimo che fa è che si ferma nel momento in cui accade. Fornisce una traccia dello stack in cui è possibile tornare indietro e trovare l'origine ...
Honey

10

Seguì il consiglio di Stephen e provò a eseguire il debug del codice e whoa! ha funzionato. La risposta sta nel messaggio di debug stesso.

Will attempt to recover by breaking constraint NSLayoutConstraint:0x191f0920 H:[MPKnockoutButton:0x17a876b0]-(34)-[MPDetailSlider:0x17a8bc50](LTR)>

La riga sopra indica che il runtime ha funzionato rimuovendo questo vincolo. Potrebbe non essere necessaria la spaziatura orizzontale sul pulsante (MPKnockoutButton). Una volta eliminato questo vincolo, non si lamenterà in fase di esecuzione e otterresti il ​​comportamento desiderato.


3
Il compilatore? Intendi il tempo di esecuzione? Il compilatore non ha rimosso il vincolo. Il compilatore l'ha lasciato lì per il runtime da gestire, quindi "ripristina rompendo il vincolo" durante il runtime .
drhr,

Sì, intendevo runtime
Sategroup

2

Ogni volta che tento di rimuovere i vincoli che il sistema ha dovuto rompere, i miei vincoli non sono più sufficienti per soddisfare l'IB (ovvero gli spettacoli "vincoli mancanti" nell'IB, il che significa che sono incompleti e non verranno utilizzati). In realtà l'ho aggirato impostando il vincolo che vuole rompere a bassa priorità, il che (e questo è un presupposto) consente al sistema di rompere il vincolo con grazia. Probabilmente non è la soluzione migliore, ma ha risolto il mio problema e i vincoli risultanti hanno funzionato perfettamente.


2
In genere si tratta di una situazione in cui si desidera utilizzare un vincolo segnaposto che viene rimosso in fase di esecuzione. Per rendere un vincolo un vincolo segnaposto, vai alla finestra di ispezione dei vincoli e fai clic su "Rimuovi al momento della creazione". Si noti come il simbolo del raggio a I del vincolo nell'area di disegno IB diventi da blu a grigio per indicarlo.
spencery2

1
Ho avuto lo stesso problema. quando rimuovo il vincolo rotto il mio design si è rotto. Quindi ho impostato la priorità su media.
Jeremy Piednoel,
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.