这是关于自动布局的第一篇文章。
>> Stack View的使用
自动布局(Auto Layout)能够根据对视图的约束(Constraint),动态地计算视图层次结构中所有视图的大小和位置。
基于约束的Auto Layout使我们能够搭建动态响应内部、外部变化的用户界面。外部变化包括用户改变窗口大小(OS X)、旋转设备(iOS)、在iPad上进入或离开分割视图(iOS)、不同屏幕尺寸,内部变化包括app显示内容长度变化、字体大小变化、对国际化的支持等。
大部分的外部变化会在运行时发生,这就要求app要动态的调整视图布局。尽管屏幕尺寸不会改变,但创建一个自动布局的界面就可以适用于iPhone SE、iPhone 7 Plus、甚至iPad不同尺寸屏幕的设备。
当用户界面中视图或控件大小变化时会导致内部变化。如对国际化的支持,把用户界面上的文字改变为其他语言时,新的语言可能占用不同大小空间;不同的语言有不同布局方向,如英语、中文都是自左向右,而阿拉伯语自右向左,这时中、英文界面右下角的按钮在阿拉伯语中应在左下角。如app支持调整字体大小,调整后用户界面中任何文本的高度和宽度都会发生变化,此时需要调整布局。
1. Auto Layout与Frame-Based Layout比较
布局用户界面方式有三种,第一种是代码方式,第二种是使用Autoresizing masks,第三种是使用Auto Layout。
通过代码方式来布局用户界面时需要设定视图在其父视图坐标系中的位置和大小。
为布局界面,你必须计算视图层级中每一个视图的位置和大小。如果其中一个发生了改变,你就需要再次计算所有受影响的视图。在许多方面,以代码方式构建的界面会更灵活、更强大。当界面有变化时,你可以操作其它视图的变化。也正因为你需要控制其它视图的变化,构建一个简单的界面可能需要大量工作,构建一个自适应的界面就变的更加困难。
第二种方式是使用Autoresizing masks来构建界面,Autoresizing masks指定视图如何随父视图的变化而改变,这简化了适应外部变化布局的复杂度。Autoresizing masks用于较小数量的可能布局,对于复杂的用户界面,需要结合使用代码布局。另外,Autoresizing masks只适用于外部变化,不响应内部变化。
第三种方式为使用Auto Layout添加一系列约束来构建界面,这些约束代表两个视图间的关系,最后Auto Layout根据这些约束计算出视图的位置和大小。Auto Layout动态的响应内部、外部变化。
视图层次结构布局被定义为一系列线性方程。每个约束表示一个方程。目标是声明一系列方程,最后只有一个可能的布局方案。下图是一个示例方程。
这个约束表示Red视图的Leading必须在Blue视图Trailing后8
points,该方程由以下几部分组成:
- Item 1:方程式中的第一项,这里指Red视图。第一项必须是视图或布局参考线(Layout Guide)。
- Attribute 1:在Item 1添加约束的位置。在这里是Red视图的Leading。
- Relationship:左右两侧的关系,可以是equal、greater than or equal和less than or equal之一。在这里左右两侧相等。
- Multiplier:第二项的值乘以该浮点值。在这里,乘数是
1.0
。 - Item 2:方程式中的第二项,在这里是Blue视图。与第一项不同,这一项可以为空。
- Attribute 2:在Item 2添加约束的位置。在这里是Blue视图的Trailing。如果Item 2为空,这里也要为空。
- Constant:浮点类型的常量。在这里是
8.0
,该值被添加到Attribute 2。
通过创建这样的方程可以创建多种约束,可以定义两个视图间的距离,对齐视图、定义两个视图大小关系、定义视图宽高比(Aspect Ratio)。
需要说明的是上面方程式中的=
是相等,不是赋值。当Auto Layout布局界面时,会计算Attribute 1和Attribute 2的值直到等式成立,而不会直接把右侧的值赋值给左侧。因此我们可以互换等式左右两侧的值,但遵守下面规则的约束更易于维护。不符合下面规则的约束可以通过互换等式两侧解决。
- Multiplier优先使用整数而非分数。
- 常量优先使用正数而非负数。
- 如果可能,视图更应该自上而下,自左向右布局。
约束既可以用来描述界面中两个视图间的关系,也可以定义一个视图中不同属性间的关系。例如设置视图高度和宽度的高宽比。
在自动布局中,可供约束的属性有左右上下四边(leading,trailing,top和bottom)、宽、高、水平居中和垂直居中等。
2. 创建demo
这里创建一个Single View Application模板的应用,Product Name为AutoLayout
,Language为Objective-C,Devices为Universal,选择文件位置,创建工程。
下面通过几个示例来体验Auto Layout。
3. 示例1
打开Main.storyboard,添加5个UILabel
。如下图所示,一个在中心,其余每个角各一个。拖放UILabel
时会出现蓝色虚线,放在这些蓝色虚线位置。
点击Xcode右上角的Assistant editor,这时一般显示的是你所选中视图控制器的代码,点击左上角的Automatic,从弹出菜单中选择Preview > Main.storyboard(Preview)。
可以通过点击上图左下角的+
按钮添加多个设备的预览,这里只添加iPhone SE。
通过上图可以看到,在4英寸的iPhone SE中,只有左上角一个Label
显示正常。这是因为如果视图没有任何约束,在Build Time系统会自动为视图添加至左上角、目前大小的约束。当你添加第一个约束时,系统会自动清除所有系统添加的约束,此时系统会出现红色警告提示约束不足。当你添加了一个约束时,你就需要添加可满足布局的所有约束。
还记得之前我们说到的蓝色虚线吗?它们不是这里说到的约束,它们是Xcode提供的布局参考线。
在Interface Builder里有三种添加约束的方式。第一种是在视图间control
+鼠标左键拖拽,也可以在Document Outline拖拽,在视图间拖拽的好处是弹出菜单会根据拖拽方向改变。第二种是使用Align和Add New Constraints工具,第三种是让Interface Builder设置约束之后我们修改约束。自动布局菜单共以下五项:
自左向右依次为:
- Update Frame:用于在约束更新后,更新frame。
- Embed in Stack:用于嵌入Stack View,会根据目前选中视图自动嵌入Horizontal Stack View或Vertical Stack View。
- Align:用于设置视图水平中心、垂直中心、基线等。
- Add New Constraints:用于设置视图与最近视图、或控制器Margin距离,视图宽、高、宽高比等。
- Resolve Auto Layout issues:添加建议约束、删除约束,其中上半部分对选中视图有效,下半部分对所有视图有效。
选中左上角Label
,点击Add New Constraints,在弹出的菜单顶部一栏左侧和上部分别填写0
和20
,点击底部Add 2 Constraints添加约束。
用同样方式为另外三个角落处的Label
添加约束,约束均是到距离自己最近的边缘,其中Leading和Trailing为0
,Top和Bottom为20
。
应尽量使用Leading
和Trailing
,避免使用Left
和Right
,这样布局会适应视图的阅读方向。阅读方向随用户设定的当前语言而变,你也可以自己设定。
最后选中视图中心的Label
,点击Align,勾选弹出窗口中Horizontally in Container和Vertically in Container选项框,后面数字均为0
,点击Add 2 Constraints添加约束。
现在进入Preview查看,一切正常。可以点击红色标记处的旋转按钮,查看横屏(Landscape)状态下Label
位置、大小。
4. 示例2
继续在刚才的demo中进行操作,打开Main.storyboard,从对象库拖拽出一个View Controller,在上面添加两个UIView
,左右各一个,左侧颜色为Blue,右侧颜色为Orange。
两个视图布局约束是:Blue视图与Top和Bottom距离是20
,与Leading距离0
,与Orange视图距离是standard。Orange视图与Top和Bottom距离是20
,与Trailing距离0
。同时Orange视图宽度是Blue视图宽度的二倍。
因为这里设置Leading、Trailing、Top和Bottom约束与前面相同,不再叙述。如果遇到问题,可以在我的Github中查看。另外,文章底部也会提供demo地址。
下面添加两个视图间间距的约束。选中Blue视图,点击Add New Constraints按钮,在弹出窗顶部点击Trailing文本框内的下三角,选择Use Standard Value,最后点击Add 1 Constraint。另外,Standard Value默认为8
。
现在添加两个视图宽度的约束。同时选中两个视图,点击Add New Constraints按钮,在弹出窗中勾选上图中的Equal Widths选项。最后点击Add 1 Constraint。如果视图出现黄色警告线,表示当前位置与视图约束位置不同,点击Update Frames更新视图位置。
现在添加Orange视图宽度为Blue视图宽度二倍的约束。选中任意一个视图,点击底部=
约束线,右侧会自动打开Attribute Inspector,鼠标放在First Item会看到选中的是Orange视图,所以这里的等式为
Orange View.Width = 2.0 x Blue View.Width
所以,修改Multiplier为2
。
在Preview查看如下,在模拟器查看效果一样。
在Add New Constraints一项中顶部有Constrain to margins选项框。如果勾选上,约束添加到父视图的margin属性;如果取消勾选,约束添加到父视图的edge属性。如果在一个控制器上添加一个UIView
,设置背景色为Red,到四边的约束均为0
,下面第一个图为勾选Constrain to margins,第二个为不勾选。想要了解详细区别,点击这里。
5. 示例3
继续对示例2中的视图进行操作,这次限定Blue视图宽度大于或等于150
。只有当屏幕足够宽时Orange视图宽度为Blue视图宽度二倍才实现,也就是可选实现。
选中Blue视图,点击Add New Constraints添加宽度为150
的约束。保持Blue视图选中状态,打开Size Inspector,双击Constraints一栏中宽度为150
的约束。也可以单击Edit按钮。
双击后弹出如下:
把上图中的Equal修改为Greater than or Equal,这样宽度就会大于等于150
。
上图中的Priority为约束优先级。所有约束默认必须实现(Required),也可以创建可选实现(Optional)的约束。
所有约束有一个优先级属性,它的值在1
和1000
间。优先级为1000
的约束必须实现,其它优先级的约束为可选实现。当自动布局计算布局方式时,会优先满足高优先级的约束。如果不能满足低优先级的约束,该约束会被跳过,但被跳过的约束不是完全无效,Auto Layout计算布局方案时会选择最接近被跳过约束的方案。
虽然优先级的值是1
到1000
,但系统把优先级划分成了低(250)、中(500)、高(750)和必需(1000)四类,你只需要把约束优先级设定为高于或低于这些值一到两个点即可。
现在storyboard出现红色警告,提示Blue视图宽度大于等于150
与Orange视图宽度为Blue视图宽度二倍冲突,把后者优先级设定为750
。Preview显示如下:
可以看到,在iPhone的竖屏(Portrait)中两个视图宽度几乎一样宽,在横屏(Landscape)中,Orange视图宽度是Blue视图宽度二倍。即优先满足了Blue视图宽度大于等于150
这一约束。
6. 示例4
目前为止,所有示例均用约束指定视图的位置和大小,但有些视图的大小根据内容变化,这称为固有内容大小(Intrinsic Content Size)。例如,UIButton
的intrinsic content size是标题大小外加很小margin。
不是所有的视图都有intrinsic content size。如:UIView
和NSView
不具有固有内容大小,UILabel
、UIButton
、UISwitch
和UITextField
高和宽都具有固有内容大小,UIImageView
没有添加图片时不具有intrinsic content size,添加图片后固有内容大小与图片大小一致。
Auto Layout在每个维度使用一对约束来表示视图的intrinsic content size。内容拥抱(Content hugging)将视图向内拉,使其紧贴内容。内容压缩阻力(Content Compression Resistance)将视图向外推,使其内容能够完整显示、不被压缩。
固有内容大小的约束有自身的优先级。一般,content hugging优先级为250
,content compression resistance优先级750
。因此,一般拉伸视图比压缩视图容易。例如:你可以把button拉伸一些,不影响使用。但如果压缩一些,内容可能无法完整显示。
继续在刚才的demo里操作,从对象库拖拽出一个View Controller,在顶部添加一个UILabel
和一个UITextField
。
添加约束,让Label与Leading距离为0
,与Text Field的Leading距离为Standard值,与Text Field基线在同一水平位置;让Text Field与Top距离20
,与Trailing距离为0
。两个视图宽和高均为固有内容大小。
这里添加约束与前面示例相同,不再叙述添加过程。只说一下添加基线约束。选中两个视图,点击Align,勾选Baselines,点击添加Add 1 Constraint。
UILabel
和UITextField
在一起时,一般UILabel
保持自身宽度,拉伸UITextField
,以此填充可用空间。所以,应当设置UILabel
的content hugging优先级为251
,UITextField
的为250
。
在Size Inspector检查content hugging,你会发现Interface Builder已经设置好了优先级。另外,如果你是用代码添加自动布局,需要手动设定这里的优先级。
点击Update Frames更新frame。在Preview查看布局。
只有当高为固有内容大小时基线才可以正确对齐。如果高被压缩或拉伸,基线将不能正确对齐。
一般在处理示例4
中的约束时应该使用Stack View。Stack View提供了一种轻松的方式利用自动布局的功能,而不会引入复杂的约束,在下一篇文章我们将详细介绍Stack View的使用。
Demo名称:AutoLayout
源码地址:https://github.com/pro648/Bas...
参考资料: