2013-10-28 4 views
10

Ich baue ein kompliziertes Projekt, wo ich unter anderem einen UIPageViewController als Kindansicht einer Hauptansicht festlegen muss. Ich verwende Autolayout und verwende Beschränkungen, um die verschiedenen Elemente in der Hauptansicht anzuordnen.Einstellung TranslatesAutoresizingMaskIntoConstraints auf Nein auf UIPageViewController

Das Problem ist, dass, wenn ich versuche, die App zu starten, es aufgrund von widersprüchlichen NSAutoresizingMaskLayoutConstraints abstürzt.

2013-10-28 16:22:18.419 Red Event App[1658:60b] Unable to simultaneously satisfy constraints. 
Probably at least one of the constraints in the following list is one you don't want. Try this: (1) look at each constraint and try to figure out which you don't expect; (2) find the code that added the unwanted constraint or constraints and fix it. (Note: If you're seeing NSAutoresizingMaskLayoutConstraints that you don't understand, refer to the documentation for the UIView property translatesAutoresizingMaskIntoConstraints) 
(
"<NSLayoutConstraint:0x145c1990 V:|-(20)-[UINavigationBar:0x145bf6b0] (Names: '|':UIView:0x145bf620)>", 
"<NSLayoutConstraint:0x145bf510 V:[UINavigationBar:0x145bf6b0]-(0)-[UIView:0x145bef70]>", 
"<NSLayoutConstraint:0x145d0550 UIView:0x145a8c10.top == UIView:0x145bf620.top>", 
"<NSAutoresizingMaskLayoutConstraint:0x145b3a40 h=-&- v=-&- UIView:0x145a8c10.midY == UIView:0x145bef70.midY + 56.5>", 
"<NSAutoresizingMaskLayoutConstraint:0x145b3a70 h=-&- v=-&- UIView:0x145a8c10.height == UIView:0x145bef70.height + 113>" 
) 

Die übliche Heilung für diese ist TranslatesAutoresizingMaskIntoConstraints auf keine Einstellung.

Allerdings, wenn ich dies tun die Child-Ansichten des UIPageViewController beginnen, die Grenzen des PageViewController zu ignorieren, und am Ende (so weit ich sehe) mit einem Ursprung von (0,0).

Ich habe versucht, die Position zu beheben, indem den Rahmen von Hand einstellen, sowohl wenn die Datenquelle (_itemViewControllers) Einrichtung:

- (void)setupLeafs{ 
    _itemViewControllers = [[NSMutableArray alloc]init]; 
    for(NSString *pagename in _datasource){ 
     RWNode *page = [_xml getPage:pagename]; 
     UIViewController *viewController = [RWNavigationController getViewControllerFromDictionary:[page getDictionaryFromNode]]; 
     viewController.view.frame = CGRectMake(self.view.frame.origin.x,self.view.frame.origin.y,self.view.frame.size.width, self.view.frame.size.height); 

     [_itemViewControllers addObject:viewController]; 
    } 
} 

und wenn die Seite

- (UIViewController *)pageViewController:(UIPageViewController *)pageViewController viewControllerBeforeViewController:(UIViewController *)viewController { 
    UIViewController *nextViewController = [self getPreviousLeaf:viewController]; 
    nextViewController.view.frame = CGRectMake(self.view.frame.origin.x,self.view.frame.origin.y,self.view.frame.size.width, self.view.frame.size.height); 
    return nextViewController; 
} 

bekommt aber weder haben irgendein Effekt.

Ich brauche die Einschränkungen für das, was ich tue, also ist das Festhalten an Masken keine Option. Ich denke, was ich suche, ist eine Möglichkeit, die UIPageViewController-Kinder, die hinzugefügt wurden (nach jedem Prozessaufruf viewControllerBeforeViewController), Einschränkungen hinzuzufügen. Aber ich würde gerne hören, wie dieses Problem gelöst werden kann.

Bearbeiten: Ich habe einen Hack gefunden, um das Problem zu lösen. Ich bin nicht ganz sicher, ob das, was ich hier aufzählen möchte, die gesamte Lösung ist, aber es ist das, was ich gerade jetzt bemerke, nachdem ich mehr als einen Monat lang an dem Code herumgebastelt habe.

Zuerst im View-Controller, der die pageviewcontroller einrichtet, habe ich die folgenden zwei Zeilen, nachdem ich die pageviewcontroller initialisiert haben

UIView *pageView = self.pageViewController.view; 
pageView.frame = self.view.frame; 

Nicht, dass ich [self.view setTranslatesAutoresizingMaskIntoConstraints:NO]; in diesem View-Controller eingestellt haben.

Zweitens, auf den untergeordneten Controllern überprüfe ich, ob die Ansicht innerhalb oder außerhalb eines Seitenaufrufcontrollers verwendet wird. Nur wenn die Ansicht nicht in einem Seitenaufrufcontroller verwendet wird, ist [self.view setTranslatesAutoresizingMaskIntoConstraints:NO]; in der untergeordneten Ansicht festgelegt.

Nun, das funktioniert (für mich) im Moment. Aber ich hätte gerne eine Lösung, die weniger hacky ist.

+0

Der Absturz ist wahrscheinlich auf etwas anderes zurückzuführen. Bei widersprüchlichen Constraints ignoriert es normalerweise einen von ihnen (und protokolliert den Fakt). Können Sie den gesamten Stack-Trace einfügen? –

+0

Es heißt (ein Beispiel zu geben): „Werden durch Zwang zu brechen zu erholen versuchen Pause auf objc_exception_throw dies im Debugger zu fangen Die Methoden in der UIConstraintBasedLayoutDebugging-Kategorie in UIView, die in aufgelistet sind, können ebenfalls hilfreich sein. " Aber darüber hinaus gibt es keine relevanten Protokollmeldungen. Und es stürzt ab, nachdem er diese Nachricht gegeben hat. – Rubberduck

+0

Können Sie beschreiben, was Sie _visuell_ erreichen möchten? Das wäre sehr nützlich, um irgendeine Art von Ratschlag zu geben :) –

Antwort

2

Ich kann Ihnen die Antwort nicht geben, aber vielleicht kann ich Ihnen einige Debug-Tipps geben .. Kopieren Sie die Protokollinformationen in einen Texteditor und ersetzen Sie die Speicheradresse Informationen für die Ansichten mit aussagekräftigen Namen. (Verwenden Sie den Debugger, um herauszufinden, welchen Eigenschaften die Speicheradressen zugeordnet sind, wenn Sie es nicht herausfinden können). Dann werden Sie es leichter finden, die Protokollinformationen zu verstehen und den Konflikt zu verfolgen. Hier ist ein Beispiel (obwohl ich bin nur Ihre Ansicht Hierarchie erraten)

"<NSLayoutConstraint:0x145c1990 V:|-(20)-[UINavigationBar:NAVBAR] (Names: '|':UIView:ROOTVIEW)>", 
"<NSLayoutConstraint:0x145bf510 V:[UINavigationBar:NAVBAR]-(0)-[UIView:PAGECONTAINER]>", 
"<NSLayoutConstraint:0x145d0550 UIView:PAGEVIEW.top == UIView:ROOTVIEW.top>", 
"<NSAutoresizingMaskLayoutConstraint:0x145b3a40 h=-&- v=-&- UIView:PAGEVIEW.midY == UIView:PAGECONTAINER.midY + 56.5>", 
"<NSAutoresizingMaskLayoutConstraint:0x145b3a70 h=-&- v=-&- UIView:PAGEVIEW.height == UIView:PAGECONTAINER.height + 113>" 

Bei einer Vermutung, die letzten beiden autoresizingmasklayoutconstraints mir seltsam aussehen, sie sehen nicht klar definiert.

+0

Mein Problem ist nicht wirklich die Einschränkungen. Ich weiß, dass es die AutoresizingMaskLayoutConstraints sind, die Probleme verursachen. Das Problem besteht darin, dass die Methode zum Entfernen von AutoresizingMaskLayoutConstraint darin besteht, TranslatesAutoresizingMaskIntoConstraints auf no zu setzen. Und wenn ich das mache, werden die Ansichten, wenn sie in einem UIPageViewController sind, vermasselt. Das ist mein Problem. – Rubberduck

6

Wenn Sie AutoLayout verwenden, sollten Sie niemals direkt den Rahmen einer Ansicht festlegen. Constraints werden verwendet, um dies für Sie zu tun. Wenn Sie in einer Ansicht eigene Einschränkungen festlegen möchten, überschreiben Sie normalerweise die Methode updateConstraints Ihrer UIViews. Stellen Sie sicher, dass die Inhaltsansichten für den Seitencontroller eine Größenanpassung der Kanten ermöglichen, da sie so angepasst werden, dass sie dem Rahmen der Seitenansicht entsprechen. Ihre Constraints und View-Setup müssen dies berücksichtigen, oder Sie erhalten unerfüllbare Constraint-Fehler.

+0

Bei der Einstellung der Frames ging es meistens darum, meine Finger zu kreuzen und zu hoffen, dass es funktionieren würde, obwohl ich wusste, dass es falsch war. Ich habe gerade versucht, Einschränkungen zwischen der Ansicht UIPageViewController und der Stammansicht des Childviewcontrollers in den updateViewConstraints des Childviewcontrollers zu setzen. Aber immer noch keine Würfel. Die Größe der untergeordneten Ansicht ist immer noch 0,0; – Rubberduck

+0

Wenn Sie Ihre View-Controller hier erstellen: 'viewController.view.frame = CGRectMake (self.view.frame.origin.x, self.view.frame.origin.y, self.view.frame.size.width, self. view.frame.size.height) ', wie groß ist die Ansicht an diesem Punkt? – gdavis

+0

Da es nicht funktionierte, und da es nichts war, was ich bei der Verwendung von Constraints tun sollte, habe ich diesen Code nicht mehr. Allerdings sehe ich meine Bearbeitung oben. – Rubberduck

4

Während ich weiß nicht genau, was Sie versuchen, visuell zu erreichen, sind hier zwei Ratschläge, die Ihnen helfen:

Erstens, denken Sie daran, alle bestehenden Beschränkungen von einem UIView vor dem Hinzufügen von neuen in entfernen Code. Mischen Sie Einschränkungen von Interface Builder nicht mit Einschränkungen im Code, es wird Sie verrückt machen. Beispiel:

- (void)viewDidLoad 
{ 
    [super viewDidLoad]; 

    // remove all constraints 
    [self.view removeConstraints:self.view.constraints]; 

    // add new constraints 
    [self.view addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[webView]|" 
                     options:0 
                     metrics:nil 
                     views:@{@"webView": self.webView}]]; 

    [self.view addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"H:|[webView]|" 
                     options:0 
                     metrics:nil 
                     views:@{@"webView": self.webView}]]; 
} 

Zweitens können Sie NSLayoutConstraint Steckdosen haben, genauso wie alle anderen Filialen von einem XIB oder ein Storyboard. Auf diese Weise können Sie bestimmte Eigenschaften einer Beschränkung zur Laufzeit anpassen:

// in your header: 
@property (nonatomic, weak) IBOutlet NSLayoutConstraint *myConstraint; 

// somewhere in your code where you need to adjust the constraint: 
self.myConstraint.constant = 100; 

Ich hoffe, dass dies ein wenig hilft :)

0

ich in einer ähnlichen Situation gerade lief. Wenn ich Einschränkungen für die Ansicht des UIPageViewControllers festlege, würde dies zu Konflikten mit den Einschränkungen der Autoresizing-Maske führen.

Was funktioniert für mich ist translatesAutoresizingMaskIntoConstraints gesetzt für UIPageController Blick auf Nein, dann Einschränkungen für die Pagecontroller Ansicht der oben, links, Breite und Höhe der übergeordneten Ansicht hinzuzufügen.

self.pageController.view.translatesAutoresizingMaskIntoConstraints = NO; 

    NSLayoutConstraint *constraint1 = [NSLayoutConstraint constraintWithItem:self.pageController.view attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:self.view attribute:NSLayoutAttributeTop multiplier:1.0 constant:0.0]; 
    NSLayoutConstraint *constraint2 = [NSLayoutConstraint constraintWithItem:self.pageController.view attribute:NSLayoutAttributeLeft relatedBy:NSLayoutRelationEqual toItem:self.view attribute:NSLayoutAttributeLeft multiplier:1.0 constant:0.0]; 
    NSLayoutConstraint *constraint3 = [NSLayoutConstraint constraintWithItem:self.pageController.view attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:self.view attribute:NSLayoutAttributeHeight multiplier:1.0 constant:-60.0]; 
    NSLayoutConstraint *constraint4 = [NSLayoutConstraint constraintWithItem:self.pageController.view attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:self.view attribute:NSLayoutAttributeWidth multiplier:1.0 constant:0.0]; 

    [self.view addConstraints:@[constraint1, constraint2, constraint3, constraint4]];