Auto Layout is kind of magic. Like a sorcerer you tell the elements where to position and how to behave. You don’t put the elements to those positions yourself. The universe moves them because of your spells. Kind of.(AutoLayout就像一種魔法.好像只要你念幾下咒語,巫師就能夠幫你把元素佈局好.)html
But when should the spells be spoken? In other words when not using Interface Builder, where should the Auto Layout code go?(問題是:在不使用IB的時候,該何時念咒語?)web
From the documentation of the UIView method updateConstraints:swift
Custom views that set up constraints themselves should do so by overriding this method.session
Because of this sentence I always thought Apple asks me to put the layout code in this method. But here is the problem: This method is called more than once by UIKit and adding the same constraint more than once is an error. The suggested approach I found in the Internet™ is to add a boolean to the view class and set it in updateConstraints() to make sure to only run the layout code once.(我覺得Apple讓我把AutoLayout的代碼放在這裏,可是在UIKit中這個方法會調用屢次,解決的辦法是添加一個布爾變量,讓這段代碼僅僅執行一次.)app
From the documentation again:less
When your custom view notes that a change has been made to the view that invalidates one of its constraints, it should immediately remove that constraint, and then call setNeedsUpdateConstraints to note that constraints need to be updated.ide
This means updateConstraints() is meant to be used in the case then the constraints within a view change because of some events. For this scenario the name of the method makes much more sense.佈局
In nearly all of my layouts the constraints are fixed. Sometimes I need to change the constant of a constraint. This can be done without removing and re-adding the constraint. (The constant of a NSLayoutConstraint is the only thing that can be changed after creation.)(改變一個約束的常數.)post
Because of this, I started to put all the layout code in init(frame:). But I was not really comfortable with it because of the documentation mentioned above.(把全部的AutoLayout的代碼都放在了init方法中.)
But then I heard in a session video of the WWDC this year an Apple engineer suggesting exactly that. And just yesterday I received an answer to a bug report in which I described the ‘bug’ that updateConstraints() isn’t called. Turns out it wasn’t a bug.
The Apple engineer wrote:
In general, if the constraints will only be created once, it should be done in an initialization method (such as -init or -viewDidLoad, and so forth). Save -updateConstraints for things that are expected to change over the course of running the App.
通常的,若是一個約束只被建立一次,這段約束的代碼應該放在初始化方法中(例如init,viewDidLoad)
Nice. I love when my feeling about how code should be written is suggested by Apple.
An example of where I put my layout code can be found here.
Happy layouting!
If you enjoyed this post, then make sure you subscribe to my feed
Update: Ole Begemann wrote about When should you implement updateConstraints on his great blog.
No related posts.
通常的,若是一個約束只被建立一次,這段約束的代碼應該放在初始化方法中(例如init,viewDidLoad)
http://swiftandpainless.com/where-to-put-the-auto-layout-code/