Why Learn about FlexBox?
I’ve heard a lot about FlexBox recently. In addition to Weex and React Native, two well-known cross-platform projects, FlexBox has also been introduced into AsyncDisplayKit.
Let’s start with the two layouts that iOS itself gives us:
- Frame, directly set the horizontal and vertical coordinates, and specify the width and height.
- Auto Layout, Layout by setting relative position constraints.
There’s not much to say about Frame, just specify the coordinates and the size, set the absolute value.
Auto Layout itself is well-intentioned, trying to free us from frames and stereotypical thinking about coordinates and sizes. Instead, use the relative position between the UIs to set the corresponding constraints for layout.
But Auto Layout’s good intentions don’t do the trick. Its syntax is too long! After two years of learning iOS, I have only used the native Auto Layout syntax for a few days. You can only use a third-party library like NAVIGATION.
The principle of Auto Layout
Let’s see how Auto Layout works.
In fact, when we set the constraints on Auto Layout, we form a set of conditions that become an equation. And then figure out the coordinates and the size of the Frame.
For example, let’s set A UI named A:
A.center = super.center
A.width = 40
A.height = 40
Copy the code
A.name = (super.center-x-40/2, super.center-y-40/2,40,40)
Set another B:
B.width = A.width
B.height = A.height
B.top = A.bottom + 50
B.left = A.left
Copy the code
B. name = (A.x, A.y + A.eight + 50, A.tip, a.eight)
As shown in figure:
Cassowary
Auto Layout has its own algorithm for dealing with constraints. I always thought it was developed by Apple, but I found out that it was from an algorithm called Cassowary.
Cassowary is an analytical toolkit for solving systems of linear equality and linear inequality, where inequality and equality are always present in the user interface. Cassowary has developed a system of rules to describe the relationships between views through constraints. Constraints are rules that indicate the position of one view relative to another.
- Dai Ming < In-depth Analysis of Auto Layout, analysis of new features of iOS versions >
Interested can further understand the implementation of the algorithm.
Frame/Auto Layout/FlexBox performance comparison
After understanding the Auto Layout, it is easy to conclude that the performance of Auto Layout is worse than that of Frame because of redundant calculation.
But by how much? What about the FlexBox?
Here, according to the test code in the performance from the Layout algorithm of Auto Layout, the Layout of Frame/Auto Layout/FlexBox is carried out, and the Layout time of 10-350 UIViews is measured in sections. The average of 100 layout times is used as the result. The unit of time is second.
The results are as follows:
While the results are inevitably biased, it’s clear from the line chart that FlexBox’s layout performance is close to that of Frame.
60 FPS, the gold standard of iOS fluency, requires layouts to be completed within 0.0166667s. The Auto Layout may start to become problematic when it has more than 50 views.
The machine configuration used in this test is as follows:
Using Xcode9.2,iPad Pro (12.9-inch)(2nd generation) emulator.
The project code for the test layout is uploaded on GitHub
What is FlexBox?
FlexBox is a UI layout that is supported by all browsers. The FlexBox is originally based on a boxed model. Flexible means Flexible, allowing it to adapt to different screens, complemishing the flexibility of the boxed model.
FlexBox treats each view as a rectangular box, with inner and outer margins, arranged along the main axis, and no dependencies between views of the same class.
Similar to Auto Layout, FlexBox uses descriptive language for Layout, unlike Frame, which uses absolute value coordinates directly for Layout.
The main idea behind a flexible layout is the ability to give Flex Container the ability to change the width and height of Flex Items to fill the available space (primarily to accommodate all types of display devices and screen sizes).
Most importantly, the FlexBox layout has nothing to do with direction, and conventional layout designs lack the flexibility to support large and complex applications (especially when it comes to direction shifts, scaling, stretching, shrinking, etc.).
FlexBox composition
The element that uses the FlexBox layout is called Flex Container.
All child elements of Flex Container are called Flex Items.
The following will talk about FlexBox inside some concepts, convenient after the use of FlexBox.
Flex Container
As mentioned earlier, one of the features of FlexBox is that there are no dependencies between views.
The Flex items are arranged depending on the properties of the Flex Container, rather than each other.
So let’s start with the Flex Containner properties.
Flex Direction
FlexBox has a main axis and a cross axis concept. The lateral axis is perpendicular to the main axis.
They can be horizontal or vertical.
The main axis defaults to Row and the side axis defaults to Column:
Flex Direction Determines the orientation of the spindles in the Flex Containner.
The main axis defaults to Row (left to right):
RowRevers can also be set (from right to left):
Column(top to bottom):
ColumnRevers(bottom to top):
Flex Wrap
The Flex Wrap determines how the view wraps when it does not align on the axis.
Flex Wrap is set to NoWrap by default, will not Wrap lines, and will line up along the main axis all the way out of the screen:
Set to Wrap to Wrap lines when there is not enough space:
If WrapReverse is set, the Wrap direction is opposite to Wrap:
This is a very useful property. For example, if iOS does not use UICollectionView to do the typical nine-grid layout, then it needs to save 9 instances, and then make a judgment, calculate the frame, the maintainability is really not high. Using UICollectionView can be a good solution to the layout, but many scenarios are not reusable, it is not particularly easy to do.
For FlexBox layouts, use the Flex Wrap property to set the Wrap.
Similar solutions on mobile platforms, such as Linear Layout on Android and UIStackView on iOS, are far less powerful than FlexBox.
Display
Display Selects whether to evaluate it. The default is Flex. If set to None, the view’s evaluation is automatically ignored.
It is useful when displaying the UI according to logic.
For example, our existing business, the need to display Tencent identity. According to the general practice, multiple ICONS are connected to each other in a row, and different distances are set according to their identities. At the same time, other ICONS are hidden, which makes comparison troublesome. The best way for iOS is to use UIStackView, which has version compatibility issues. With the FlexBox layout, when not an identity, as long as Display is set to None, it will not be included in the UI calculation.
Justify Content
The text Content defines the alignment of Flex items on the spindle :FlexStart, FlexEnd, and Center.
And SpaceBetween (align both ends) :
Set the alignment on both ends so that the Spacing between Flex items is equal.
SpaceAround(equal margins) :
Make the margins equal around each Flex Item
Align Items
Align Items define how Flex Items are aligned on the side axis.
Can Align the Items and the main shaft alignment the Justify the Content, set FlexStart, FlexEnd, Center, SpaceBetween, SpaceAround.
Align Items with Baseline:
As shown, it is a baseline alignment based on the first line of text of the Flex Item.
If the Baseline and Flex items have the same inline and side axes, this value is equivalent to FlexStart. In other cases, this value participates in baseline alignment.
Align Items can also be set to Stretch:
Stretch makes the Flex Item Stretch and fill the entire Flex Container. Stretch makes the margin of a Flex Item as close to the size of the row or column as possible, following the corresponding property limits.
If Flex Item is not set, or set to Auto, it will take up the entire height of The Flex Container
Align Content
Align Content is also the alignment of the side axis in the Flex Item, but with the entire line as the minimum unit.
Note that this property does not work if the Flex Item has only one axis (a one-line Flex Itme).
Change FlexWrap to Wrap to display the effect:
Flex Item
The properties of a Flex Container apply to its own contained Flex Item. The properties of a Flex Item apply to itself.
AlignSelf
AlignSelf allows a single Flex Item to be aligned differently from other Flex Items, overriding the Align Items property.
The default value is Auto, which means that the Align Items property of Flex Container is inherited. If it does not have a Flex Container itself, it is equivalent to Stretch.
FlexGrow
FlexGrow allows you to set the proportion of free space allocated. How to expand.
The default value of FlexGrow is 0. If FlexGrow is not defined, the layout does not have the right to allocate any remaining space.
Such as:
If the overall width is 100, the width of sub1 is 10, and the width of sub2 is 20, the remaining space is 70.
Set FlexGrow to allocate this ratio of 70 widths.
Again, the question of proportion value:
If all Flex items have a FlexGrow attribute of 1, the remaining space, if any, is divided equally.
If one Flex Item has a FlexGrow attribute of 2 and all other Flex items are 1, the remaining space of the Flex Item will be twice that of the other Flex items.
FlexShrink
In contrast to FlexGrow, which deals with remaining space, FlexShrink is used to handle situations where there is not enough space. How to shrink.
FlexShrink defaults to 1, meaning that the project will shrink if space runs out
If all Flex items have a FlexShrink property of 1, they will all be scaled down equally when space runs out.
If one Flex Item has a FlexShrink property of 0 and all other Flex items have a FlexShrink property of 1, then the former Item where FlexShrink is 0 is not shrunk when space is short.
FlexBasis
FlexBasis defines the main size that a Flex Item takes up before any extra space is allocated. The browser uses this property to calculate whether the main axis has extra space.
The default value of FlexBasis is Auto, which is the original size of the Flex Item.
To learn more about FlexBox properties, see A Complete Guide to FlexBox
FlexBox implementation – Yoga
As mentioned earlier, the FlexBox layout has been used by several well-known open source projects, including Yoga from Facebook.
Yoga is a Flexbox layout engine implemented by C, and its performance and stability have been well proven in various major projects. However, Yoga only implements a subset of the W3C standard.
Here’s a look at YogaKit, the implementation of Yoga on iOS.
Based on the basic understanding of FlexBox layout above, make some simple layouts.
YGLayout
The whole point of YogaKit is in the YGLayout object. YGLayout is used to set the layout properties.
In the UIView+ yoga.h file:
/**
The YGLayout that is attached to this view. It is lazily created.
*/
@property (nonatomic, readonly, strong) YGLayout *yoga;
/**
In ObjC land, every time you access `view.yoga.*` you are adding another `objc_msgSend`
to your code. If you plan on making multiple changes to YGLayout, it's more performant to use this method, which uses a single objc_msgSend call. */ - (void)configureLayoutWithBlock:(YGLayoutConfigurationBlock)block NS_SWIFT_NAME(configureLayout(block:));Copy the code
You can see a called yoga YGLayout read-only objects, and configureLayoutWithBlock (YGLayoutConfigurationBlock) block method, NS_SWIFT_NAME() is also used to define the method name in Swift.
So we can use the UIView instance object directly, to set its layout directly.
isEnabled
That’s how isEnabled is defined in yglayout.h.
/**
The property that decides during layout/sizing whether or not styling properties should be applied.
Defaults to NO.
*/
@property (nonatomic, readwrite, assign, setter=setEnabled:) BOOL isEnabled;
Copy the code
IsEnabled defaults to NO, and we need to set it to YES during the layout to enable the Yoga style.
applyLayoutPreservingOrigin:
For this method, the header file reads:
/**
Perform a layout calculation and update the frames of the views in the hierarchy with the results.
If the origin is not preserved, the root view's layout results will applied from {0,0}.
*/
- (void)applyLayoutPreservingOrigin:(BOOL)preserveOrigin
NS_SWIFT_NAME(applyLayout(preservingOrigin:));
Copy the code
In simple terms, it is used to perform layout calculations. So, once the layout code is complete, call this method on the root view property yoga object and apply the layout to the root view and its subviews.
Layout of the presentation
Here is an example of how to use Yoga for FlexBox layout.
centered
[self configureLayoutWithBlock:^(YGLayout * layout) {
layout.isEnabled = YES;
layout.justifyContent = YGJustifyCenter;
layout.alignItems = YGAlignCenter;
}];
[self.redView configureLayoutWithBlock:^(YGLayout * layout) {
layout.isEnabled = YES;
layout.width=layout.height= 100;
}];
[self addSubview:self.redView];
[self.yoga applyLayoutPreservingOrigin:YES];
Copy the code
The effect is as follows:
Our actual layout code simply justifys the Flex Container’s contentations and alignItems.
Nested layout
Make a view slightly smaller than its superView with a margin of 10:
[self.yellowView configureLayoutWithBlock:^(YGLayout *layout) {
layout.isEnabled = YES;
layout.margin = 10;
layout.flexGrow = 1;
}];
[self.redView addSubview:self.yellowView];
Copy the code
The effect is as follows:
The layout code only sets the margin and flexGrow of the View.
Equispaced arrangement
A group of views arranged with equal spacing vertically:
[self configureLayoutWithBlock:^(YGLayout *layout) {
layout.isEnabled = YES;
layout.justifyContent = YGJustifySpaceBetween;
layout.alignItems = YGAlignCenter;
}];
for( int i = 1 ; i <= 10 ; ++i ) { UIView *item = [UIView new]; Item. BackgroundColor = [UIColor colorWithHue:(arc4random() % 256/256.0) saturation:(arc4random() % 128/256.0) + Sparent (sparent () % 13/256.0) + 0.5alpha :1]; [item configureLayoutWithBlock:^(YGLayout *layout) { layout.isEnabled = YES; layout.height = 10*i; layout.width = 10*i; }]; [self addSubview:item]; }Copy the code
The effect is as follows:
This can be done easily by setting The Layout. Contusions ycontent = YGJustifySpaceBetween of Flex Container.
Equal spacing, automatic set width
Let the two views of height 100 be vertically centered, equal width, equal spacing, 10 apart. Automatically calculate its width:
[self configureLayoutWithBlock:^(YGLayout *layout) {
layout.isEnabled = YES;
layout.flexDirection = YGFlexDirectionRow;
layout.alignItems = YGAlignCenter;
layout.paddingHorizontal = 5;
}];
YGLayoutConfigurationBlock layoutBlock =^(YGLayout *layout) {
layout.isEnabled = YES;
layout.height= 100;
layout.marginHorizontal = 5;
layout.flexGrow = 1;
};
[self.redView configureLayoutWithBlock:layoutBlock];
[self.yellowView configureLayoutWithBlock:layoutBlock];
[self addSubview:self.redView];
[self addSubview:self.yellowView];
Copy the code
The effect is as follows:
We just need to set paddingHorizontal of Flex Container and marginHorizontal of Flex Item and flexGrow. You can reuse the Layout style of the Flex Item.
UIScrollView alignments automatically compute contentSize
Arrange some views in UIScrollView order and automatically calculate contentSize:
[self configureLayoutWithBlock:^(YGLayout *layout) {
layout.isEnabled = YES;
layout.justifyContent = YGJustifyCenter;
layout.alignItems = YGAlignStretch;
}];
UIScrollView *scrollView = [[UIScrollView alloc] init] ;
scrollView.backgroundColor = [UIColor grayColor];
[scrollView configureLayoutWithBlock:^(YGLayout *layout) {
layout.isEnabled = YES;
layout.flexDirection = YGFlexDirectionColumn;
layout.height =500;
}];
[self addSubview:scrollView];
UIView *contentView = [UIView new];
[contentView configureLayoutWithBlock:^(YGLayout * _Nonnull layout) {
layout.isEnabled = YES;
}];
for( int i = 1 ; i <= 20 ; ++i ) { UIView *item = [UIView new]; Item. BackgroundColor = [UIColor colorWithHue:(arc4random() % 256/256.0) saturation:(arc4random() % 128/256.0) + Sparent (sparent () % 13/256.0) + 0.5alpha :1]; [item configureLayoutWithBlock:^(YGLayout *layout) { layout.isEnabled = YES; layout.height = 20*i; layout.width = 100; layout.marginLeft = 10; }]; [contentView addSubview:item]; } [scrollView addSubview:contentView]; [scrollView.yoga applyLayoutPreservingOrigin:YES]; scrollView.contentSize = contentView.bounds.size;Copy the code
The effect is as follows:
The main use of the UIScrollView is an intermediate contentView, which calculates the contentSize of the ScrollView. To note here is that to finish in scrollview call applyLayoutPreservingOrigin: after set, otherwise can not get the result.
The use of UIScrollView, currently on the Internet also did not find a more official example, completely is the author’s own groping, welcome to know the big man to teach.
The sample code used above has been uploaded to GitHub
conclusion
FlexBox is a very good layout for mobile, with clear meaning and stable performance. As mobile UI views become more complex, it is important for mobile developers to understand new solutions, especially as FlexBox is now supported in all browsers.
After you become familiar with the YogaKit approach, you can also try to encapsulate a set of layout code to speed up development efficiency.
Reference:
Flex Layout tutorial: Syntax section
FlexBox layout model
YogaKit
Yoga Tutorial: Using a Cross-Platform Layout Engine
On the performance of Auto Layout algorithm