Disposable UI with the Interface Builder and Autolayout
Bohdan Orlov
iOS Developer @Badoo
Reusable UI
- customisable
- written in a code
- very generic
- built for the future
- requires a lot of time
Disposable UI
- not reusable
- made in the Interface Builder
- very specific
- built to be scrapped
- fast to build
Disposable UI
- not all tasks can be done in the IB
- logic is spread between UI files and code
- Its fast - less code has to be written.
- less effort put to IB files so its easier to say "good bye"
- One glance to IB file gives general idea of UI
- able to achieve anything
- everything in one place
- code has to be written
- code has to be supported
- code will be scrapped and you will feel bad, its only matter of time
- UI is "invisible" until whole code is read
In a code
In the IB
Building disposable UI in the IB
- Reasons to use Autolayout
- Common issues adopting Autolayout in the IB
- Getting the most of IB using Size Classes
- Principle of least astonishment (surprise) using the IB
- IB tips and tricks
- Working with Autolayout in a code
- Debugging Autolayout
Agenda
Reasons to use Autolayout
- We can't hardcode frames in the IB anymore
- We have system wide font settings (Dynamic Type)
- Internationalisation (right to left)
- Layout is expected to be adaptive (more than a simple stretching)
- Autosized cells (if performance is acceptable)
- WatchKit apps require Storyboards.
To memory of -setFrame: and CGRectDivide()
data:image/s3,"s3://crabby-images/2eefb/2eefba62879980e5d8d95aeec858ac97846984a4" alt=""
Common issues adopting Autolayout in the IB
- Ambiguous constraints
- Constraint priorities
- Constraints to Margins
- Equal space between views
- Views group centring
- Intrinsic content size
- Multiline UILabel
- Content hugging priority
- Content compression resistance priority
Ambiguous constraints
- Don't press "Add Missing constraints" button
- understand the reason
- add constraints manually
data:image/s3,"s3://crabby-images/dd52c/dd52c0e079d6938088904f8668fff790ab888198" alt=""
Red circle:
Yellow triangle:
- "Update frames" is our choice
- Other options might update your constraints in unexpected way
data:image/s3,"s3://crabby-images/55e64/55e64686227164a96458f5139c7f3f245f61a20d" alt=""
Getting rid of:
Ambiguous constraints
Adding constaraints manually
data:image/s3,"s3://crabby-images/37f33/37f33fa09a9863f4c45f41ddfd8ef2634b00abfd" alt=""
There are multiple ways to achieve same result with different constraints
Inequality constraint ambiguity
data:image/s3,"s3://crabby-images/89133/8913319b1bbb7fb0f4ba84e695a6808a514913a8" alt=""
We can resolve this by adding additional constraints with priority less than required.
For example we can say:
View.width = width @ 750
Constraint priorities
static const UILayoutPriority UILayoutPriorityRequired NS_AVAILABLE_IOS(6_0) = 1000;
static const UILayoutPriority UILayoutPriorityDefaultHigh NS_AVAILABLE_IOS(6_0) = 750;
static const UILayoutPriority UILayoutPriorityDefaultLow NS_AVAILABLE_IOS(6_0) = 250;
// the priority level with which the view wants to conform to the target size.
static const UILayoutPriority UILayoutPriorityFittingSizeLevel NS_AVAILABLE_IOS(6_0) = 50;
data:image/s3,"s3://crabby-images/f1998/f1998ba31254f0572d4311528013080362ff93f8" alt=""
- Its possible to have multiple constraint for same views/attributes
- First constraint is not required thus if there is not enough space it won't be satisfied
Constraints to Margins
data:image/s3,"s3://crabby-images/62bc8/62bc8c867a5b507e9afc7cc0c3db6260d5b22f24" alt=""
data:image/s3,"s3://crabby-images/faf69/faf69c5c47a240173362f62d232be9444f0516a2" alt=""
data:image/s3,"s3://crabby-images/5c32b/5c32b5f688a480c0675dd085cb0bea02363e17e6" alt=""
data:image/s3,"s3://crabby-images/9f752/9f7529fca8e651ca634dfe1ca373fba3b58656f4" alt=""
Equal space between views
data:image/s3,"s3://crabby-images/9a7cb/9a7cbf27cb0989ef2d688762d3c52f140b4e4342" alt=""
data:image/s3,"s3://crabby-images/1a1a2/1a1a243a348480337f2c98c985e90a592a569277" alt=""
For iOS 9+ we should use UIStackView
Constraints can be setup only on view attributes, but not other constraints.
We have to use dummy spacing views, with equal width constraint and 0 leading/trailing to neighbours
Views group centring
For iOS 9+ we should use UIStackView
Constraints can be setup only on view attributes, but not other constraints.
We have to use dummy container view
data:image/s3,"s3://crabby-images/a01d6/a01d61d471737f597cd81f9aa7fe1c08c6dd8079" alt=""
data:image/s3,"s3://crabby-images/cbb3f/cbb3fa6b254ec90becdf60f560b7820503525807" alt=""
Intrinsic content size
- Content-defined size which allows to skip setting explicit constraint for width and/or height
- UILabel and UIButton have intrinsic content size for both width and height
-
Override for custom/frame based views, invalidate it when appropriate
-
You can use UIViewNoIntrinsicMetric for width or height
-
Intrinsic content size of a view refers to its alignment rect, not to its frame.
data:image/s3,"s3://crabby-images/5e772/5e772ee9e591b8e780d9c29104d721faaa205ff7" alt=""
data:image/s3,"s3://crabby-images/caf89/caf8923a58d6007d90caeece80d2fbc834344577" alt=""
Multiline UILabel
-
The intrinsic content size of multiline UILabel depends on the width of the lines, which is yet to be determined when solving the constraints.
- (void)layoutSubviews {
[super layoutSubviews];
myLabel.preferredMaxLayoutWidth = myLabel.frame.size.width;
[super layoutSubviews];
}
- preferredMaxLayoutWidth, specifies the maximum line width for calculating the intrinsic content size.
- Supposed to work without extra code in iOS 8+
data:image/s3,"s3://crabby-images/dcbca/dcbcac30989ef20f185e2e94c7b6086284333bc7" alt=""
Content hugging priority
data:image/s3,"s3://crabby-images/c89ce/c89cebf0cb5bae2ab00e9ed6a1c4068147350274" alt=""
Content compression resistance priority
data:image/s3,"s3://crabby-images/0c118/0c1184a940e9b3e8778260f74463826b9239745a" alt=""
- Only take effect for views which define an intrinsic content size.
- Translated into constraints implicitly
H:[label(<=100@250)]
V:[label(<=30@250)]
H:[label(>=100@750)]
V:[label(>=30@750)]
Getting the most of IB using
Size Classes
data:image/s3,"s3://crabby-images/89a01/89a014ee2b3676f07ad7ae11bbd92a44bae7352a" alt=""
data:image/s3,"s3://crabby-images/ff6ca/ff6ca0d704a6aa77282b8a5e0a5d2d4ff05e7705" alt=""
Size Classes depends on:
- device
- orientation
- special cases like split view
data:image/s3,"s3://crabby-images/226e0/226e046600aeb68f976ea309c74a56044e1aef58" alt=""
<- iPhone 6 Plus is exception
iPhones' Size Classes
Selecting a Size Class in
Interface Builder
data:image/s3,"s3://crabby-images/bd486/bd48681da2bce45e410996b85215bf7c98572704" alt=""
data:image/s3,"s3://crabby-images/fc882/fc882990437dcd35975cc236de0ae6b832c1e9ff" alt=""
data:image/s3,"s3://crabby-images/aeb4b/aeb4b4ac22c78a34355b6d1e52e9e5c69793dd8d" alt=""
- Changing size or position of views
-
Adding or removing views
-
Adding or removing constraints
-
Changing the font in labels, fields, text views, and buttons
Changes available for different size classes
data:image/s3,"s3://crabby-images/c91bf/c91bfab048c67a5fb8b9ba71ccad58cfd4ed48f7" alt=""
data:image/s3,"s3://crabby-images/e7952/e79527b136a4a98ab196431c7f61be410ec7be41" alt=""
Principle of least astonishment using the IB
data:image/s3,"s3://crabby-images/e6320/e6320435ba0da495c5ac73a46638fd18c7be0cd2" alt=""
Counterexamples
data:image/s3,"s3://crabby-images/34d1e/34d1e059496b92bfe5b1df83cc1e555b89ca2e71" alt=""
data:image/s3,"s3://crabby-images/4e498/4e49853b7d9de15d2d329ba8af12535c27c04d9c" alt=""
Principle of least astonishment using the IB
Examples
data:image/s3,"s3://crabby-images/d6133/d6133d2560e9a7db0cea1d98ed25569692caffbc" alt=""
data:image/s3,"s3://crabby-images/e4d97/e4d970fa01db7a21caaaa6c4343fcec1f037a301" alt=""
data:image/s3,"s3://crabby-images/98983/98983b813bedfbbed1b9c52f40242b150aeaa9b1" alt=""
Dos:
- Put dummy values into labels/buttons
- Put dummy background color
- Put dummy image into UIViews
- Use IBDesignable and IBInspectable
Don'ts:
- Don't use User Defined Runtime Attributes
- Don't override values in code with same value.
- Avoid removing views at runtime, prefer deactivating constraints
data:image/s3,"s3://crabby-images/00754/007549e279f3e64c00d60ddb44f8ed09173df3d3" alt=""
data:image/s3,"s3://crabby-images/d9c81/d9c819fb5c88642b673bfbb1dcb92d7afc3488bd" alt=""
Principle of least astonishment using the IB
IB tips and tricks
data:image/s3,"s3://crabby-images/4dc41/4dc4108818dcc8d97151a3597290f4d07c440a4b" alt=""
data:image/s3,"s3://crabby-images/1dc81/1dc814a44ceb2fc3147d5300943941103b0530f5" alt=""
data:image/s3,"s3://crabby-images/7924f/7924f1ccf0d0031ff684a7eb6fe75c9abcea0065" alt=""
data:image/s3,"s3://crabby-images/6e9c3/6e9c3cfa00f95fee44653f37552677c26478f2e1" alt=""
data:image/s3,"s3://crabby-images/0c4b3/0c4b37817e195c03f634e98e86574f77459b0757" alt=""
SHIFT + OPT + LMB
SHIFT + CTRL + LMB
IB tips and tricks
^ + OPT
Top level objects in Scene Dock
data:image/s3,"s3://crabby-images/010a9/010a9e41f1ab2b00ed2191e46a5b5a8e082d4aba" alt=""
data:image/s3,"s3://crabby-images/629a5/629a53118913830faed88faba88d9b7130222595" alt=""
data:image/s3,"s3://crabby-images/ee285/ee285e0a85d044562c78e760acf3d152301eb735" alt=""
data:image/s3,"s3://crabby-images/53976/539767fa27d93e2897c254df4a66c52f79976c80" alt=""
Drag with CTRL to view
<+ SHIFT
to add multiple
data:image/s3,"s3://crabby-images/b7da1/b7da147c405a93b66a8252ad31dadd18dc9d6c25" alt=""
^ LMB + LMB
IB tips and tricks
data:image/s3,"s3://crabby-images/c982a/c982a5b6814f56a30357349f0825cc1cbe3a0b6e" alt=""
@IBInspectable / @IBDesignable
IB tips and tricks
Simulated metrics
data:image/s3,"s3://crabby-images/6edb8/6edb8da0e18413c0cda920f643045933eb712520" alt=""
Strong Outlets (again)
data:image/s3,"s3://crabby-images/09d78/09d78f41075df60742a3a42554c6d22aed803772" alt=""
Сanvas configuration
data:image/s3,"s3://crabby-images/8229b/8229bff714b3e6361c3c6070ab23e98ad79a1083" alt=""
data:image/s3,"s3://crabby-images/0ddde/0ddde6eda9264dab5585a089406712fa6f89d711" alt=""
Multiple nav bar items
IB tips and tricks
Сanvas configuration
data:image/s3,"s3://crabby-images/8229b/8229bff714b3e6361c3c6070ab23e98ad79a1083" alt=""
data:image/s3,"s3://crabby-images/0ddde/0ddde6eda9264dab5585a089406712fa6f89d711" alt=""
Multiple nav bar items
Working with Autolayout
in a code
- Don't work with Autolayout in code
- Don't add constraints in code
- Don't be naive about Visual Format Language
- Don't apply transform to views with top/left/bottom/right constraints
- Don't change constraints inLayoutSubviews
- NSLayoutConstraint has only one* mutable property - constant.
- Activate/deactivate constraints instead of adding/removing
- Minimise amount of constraints with outlets
- Beware of what you animating
NSLayoutAnchor/UILayoutGuide available since iOS 9
Working with Autolayout
in a code
Animation
// Ensures that all pending layout operations have been completed
[containerView layoutIfNeeded];
[UIView animateWithDuration:1.0 animations:^{
// Make all constraint changes here
// Forces the layout of the subtree animation block and captures all of the frame changes
[containerView layoutIfNeeded];
}];
Opting out from Autolayout
override func layoutSubviews() {
mySubview.frame = CGRectMake(...)
}
Working with Autolayout
in a code
- (void)updateConstraints {
[self removeConstraints:self.imageViewConstraints];
self.imageViewConstraints = [self createConstraints];
[self addConstraints:self.imageViewConstraints];
[super updateConstraints];
}
Updating/Adding constraints
Baseline alignment
- (UIView *)viewForBaselineLayout
{
// create the view if not exists, start with rect zero
if (!self.baselineView) {
...
}
int integerBaseline = ...
// update the frame of the view
self.baselineView.frame = CGRectMake(0, 0, self.bounds.size.width, (float)integerBaseline);
return self.baselineView;
}
Working with Autolayout
in a code
activateConstraints: instead of addConstraints:
translatesAutoresizingMaskIntoConstraints
- YES for any view you programmatically create.
- NO If you add views in Interface Builder
- Automatically adds the constraints to the correct view.
systemLayoutSizeFittingSize:
- CGSize UILayoutFittingCompressedSize
- CGSize UILayoutFittingExpandedSize
- UILayoutPriorityFittingSizeLevel = 50
Working with Autolayout
in a code
Alignment rects
- - alignmentRectForFrame:
- - frameForAlignmentRect:
- - alignmentRectInsets
data:image/s3,"s3://crabby-images/be97b/be97b9d5e9f256b342c0222adb9df3dd2e7f279d" alt=""
Debugging Autolayout
Use identifiers
- Constraints identifiers
- Views identifiers
- UILayoutGuides identifiers
Debugging methods
- constraintsAffectingLayoutForAxis:
- hasAmbiguousLayout
- exerciseAmbiguityInLayout
*<UIWindow:0x7fe781d25a80> - AMBIGUOUS LAYOUT
| *<UIView:0x7fe781d29db0>
| | *<UIButton:0x7fe781d2a170>
| | | <UIButtonLabel:0x7fe781c107d0>
| | *<UISlider:0x7fe781d2b9b0>
| | | <UIView:0x7fe781d37fe0>
| | | | <UIImageView:0x7fe781d3a780>
| | | <UIImageView:0x7fe781d3ac50>
| | | <UIImageView:0x7fe781d3e590>
| | | | <UIImageView:0x7fe781d39a00>
| | *<_UILayoutGuide:0x7fe781d2c4b0> - AMBIGUOUS LAYOUT
| | *<_UILayoutGuide:0x7fe781d2cda0> - AMBIGUOUS LAYOUT
expr ((UIView *)0x7731880).backgroundColor = [UIColor purpleColor]
po [[UIWindow keyWindow] _autolayoutTrace];
Storyboards
- Merges
- No DI via initialiser
- Segues (Unwind segues)
- Space to grow your screen naturally
- Refactoring with Storyboard References
DEMO
IB
By borlov
IB
- 1,398