Auto layout invalid after presenting view controll

2020-02-07 20:27发布

Background:

I have a custom container controller, which just like UITabBarController, but with switch animation. And I use auto layout to achieve.
Because switch between child controllers is dynamic, the proper constraints are added to child view controller when its's view is added into container's view, not pre-setup in IB. (The constraints are added to the superview of course)

http://d.pr/i/q6NF Container Controller Nib Setup

PS: The detail of constraints
H:|[Child](Change the constraint's constant to animation from left/right to right/left)
H:[Child(==Super)]
V:|[Child]|

Where is the Problem:

One of the child controller is a navigation controller, things go wrong when the navigation controller present a modal view controller (using presentViewController:animated:completion:), and dismiss it(usingdismissViewControllerAnimated:completion:), after present/ dismiss the frame of navigation controller become (x, y, 0, 0), It seems like the auto layout becomes invalid, maybe the constraints been removed.

http://d.pr/i/VmvL The Present/ Dismiss Process

Research

I haven't use code to verify what happen to these constraints yet, but with Spark Inspector I see the views presentation change during the present/ dismiss process. When My navigation controller present a modal view controller, iOS just swap the whole navigation controller's view to the modal view controller's view. And when navigation controller's view come back, the auto layout doesn't work anymore.

Possible Solution

One of the solution I came up with is let my container controller present modal controller.

Or I just change my container controller to no auto layout.

The fact is ever since I start using auto layout, the problems this technique causes just dominate the benefits. Besides this problem, Every time the interface orientation changes, views inside my container controller just can't auto layout right, it seems subviews always use the frame of superview before orientation change. I double check the constraints I setup, there is no conflicts and no ambiguity.

My guess is that my custom container controller isn't compatible with presenting modal view controller and interface orientation change in Auto Layout system, even with constraints setup.

Xcode Environment

Xcode 5 beta, iOS 7SDK, target iOS6.1 Maybe Something not right with the SDK environment?

6条回答
beautiful°
2楼-- · 2020-02-07 20:46

I had the same problem, with a search bar container as the misplaced element (inside a UIViewController, inside a UITabBarController). None of the other solutions I've seen worked, but I eventually managed to work around it by re-adding the layout constraint in the viewWillAppear:

[self.view addConstraint:[NSLayoutConstraint constraintWithItem: self.searchBar.superview
                                                      attribute: NSLayoutAttributeTop
                                                      relatedBy: NSLayoutRelationEqual
                                                         toItem: self.view
                                                      attribute: NSLayoutAttributeTop
                                                     multiplier: 1
                                                       constant: 0]];
查看更多
女痞
3楼-- · 2020-02-07 20:48

I have also the same problem and when i tried this navigation controller it was working fine but not with present viewcontroller. Use this below view controller method along with translatesAutoresizingMaskIntoConstraints to solve this problem.

-(void) viewWillLayoutSubviews
{
    [super viewWillLayoutSubviews];

    self.view.translatesAutoresizingMaskIntoConstraints = NO;
}

Please let me now it you have any concern about it. Thanks

查看更多
Viruses.
4楼-- · 2020-02-07 20:48

I had a similar problem with table view cells that I was laying out with autolayout. Upon returning from a modal view, the cells had an invalid layout. I could re-layout each cell after viewDidAppear, but it looked terrible. Thanks to @palimondo's suggestion I started sniffing around translatesAutoresizingMaskIntoConstraints. It turns out that I was setting the content view of my table view cell to translatesAutoresizingMaskIntoConstraints = NO when I shouldn't have been.

查看更多
姐就是有狂的资本
5楼-- · 2020-02-07 20:51

It is probably caused by ambiguous constraints. I recommend pause a running application and type a following command to console:

po [[UIWindow keyWindow] _autolayoutTrace] 
查看更多
beautiful°
6楼-- · 2020-02-07 20:56

I had a similar problem. I was setting translatesAutoresizingMaskIntoConstraints = NO; on my root UIView. It appears the "outermost" UIView - the superview at the root of your hierarchy must use the default translatesAutoresizingMaskIntoConstraints = YES. Once I've removed this, everything worked as expected.

查看更多
萌系小妹纸
7楼-- · 2020-02-07 20:59

If you use auto layout between the UIWindow and your root view (to set the root view to fill the entire UIWindow) these constraints will get blown away by a full screen presentation of another view controller.

What happens, is the entire hierarchy of the full-screen presentation replaces everything below UIWindow - your original view is removed (blowing away the constraints) and the new view hierarchy substituted. Then when your review is replaced, these constraints are lost. You would need to recreate them somewhere like viewWillAppear: or just ensure your root view has self.view.translatesAutoresizingMaskIntoConstraints = NO;

查看更多
登录 后发表回答