注:本文节选并翻译自Anatomy of a Constraint
真实内容尺寸
到目前为止,所有的示例都是用约束来定义视图的位置和尺寸的。可是,某些视图会根据当前内容生成出一个自然尺寸。这个尺寸即可参考着叫做真实内容尺寸。比如,一个按钮的真实内容尺寸就是它的标题加上一个小边缘。
不是所有的视图都有真实内容尺寸,对于普通视图来说,真实内容尺寸可以定义出视图的宽、高或者二者全部。下面的表格3-1中列出了一些示例。
- 表格3-1 基本控件的真实内容尺寸
视图 | 真实内容尺寸 |
---|---|
UIView和NSView | 无 |
Sliders | 只定义了宽度(iOS) |
Labels、buttons、switches和text fields | 同时定义了宽度和高度 |
Text views和image views | 不同情况下差别很大 |
真实内容尺寸基于视图的当前内容。一个label或button的真实内容尺寸基于显示文字的总量和使用的字体。对于其他视图来说,真实内容尺寸可能会更复杂。例如,一个空的image view没有真实内容尺寸。一旦在上面添加图片之后,它的真实内容尺寸就被设置为图片的尺寸了。
一个text view的真实内容尺寸有多种不同情况,取决于其内容、是否可以滑动或者是给它添加的其他约束。比如,当可以滑动时,text view没有真实内容尺寸。当不可滑动时,默认来说视图的真实内容尺寸基于不换行的文字计算而来。再例,在文字中如果不存在换行,真实内容尺寸会计算出需要的宽和高,并将内容布局为一个单行文字。如果你添加约束指定了text view的宽,真实内容尺寸就会根据此宽度来定义出需要的高度并进行显示。
Autolayout通过对每一个尺寸使用一对约束来表示真实内容尺寸。抗拉伸约束(content hugging,内容包裹)把视图向内部推以保证它可以优雅地环抱着其内容。抗压缩约束(compression resistance)把视图向外推以便它不会轻易裁剪其内容。
在列表3-5中展示的这些约束通过使用不等式来进行定义。这里,IntrinsicHeight和IntrinsicWidth常量代表了视图真实内容尺寸中的宽和高。
- 列表3-5 抗压缩和抗拉伸约束方程
|
|
这些约束中的每一个都有自己的优先级。默认来说,视图使用250优先级来定义抗拉伸约束,而使用750优先级来定义抗压缩约束。因此,对于压缩视图来说,拉伸它更加容易。对大多数的控件来说,这都是默认行为。例如,你可以安全地将一个button拉伸到大于它本身的真实内容尺寸;但是,如果你压缩它,则很可能会裁剪其内容。记住,Interface Builder可能会适当修改这些优先级以避免此种情况发生。了解更多信息,请看设置抗拉伸和抗压缩约束的优先级。
尽可能地在布局中多使用视图的真实内容尺寸。它可以让你的布局动态适配视图内容的变化。它还可以减少约束的数量来创建出明确的、无冲突的布局,但是你需要管理该视图的抗拉伸和抗压缩约束(统称为CHCR)的优先级。这里是一些处理真实内容尺寸的指导建议:
- 当拉伸一些列视图来填充空间时,如果所有视图都有相同的抗拉伸优先级,此布局就存在歧义了。Autolayout不知道应该拉伸哪个视图。
一个普遍的例子就是label和text field的组合。一般来说,你想要让text field来拉伸填充额外空间,而label保持它自身的真实内容尺寸。要确保这种情况,要保证text field的水平抗拉伸优先级要低于label。
实际上,这个例子如此普遍,Interface Builder已经可以自动处理此情况,自动将label的抗拉伸优先级设置为251。如果手动编码进行布局,你需要自己修改抗拉伸优先级。 - 当一些带有不可见背景的视图(如buttons或labels)被拉伸超过本身真实内容尺寸时,可能经常会显示出怪异或非预期的布局。导致的实际问题可能并不严重,因为文字直接出现在了错误位置上。要避免这种不必要的拉伸,增加抗拉伸优先级即可。
- 基准线约束只在那些存在真实内容高度的视图上才有效。如果一个视图被垂直拉伸或压缩,基准线约束便不再正常匹配。
- 一些视图,如switches,应该总是以真实内容尺寸进行显示。通过提高CHCR优先级来防止被拉伸或压缩。
- 避免给视图设置“必须级”CHCR优先级。对于视图来说错误的显示尺寸要比偶然出现的约束冲突要更好。如果一个视图需要一直保持真实内容尺寸,考虑将其设置为“非常高”(999)优先级。这个接近“满级”的优先级统称可以保证视图不被拉伸或压缩,而且还可以提供出一个紧急值,以免你的视图在一个与你期望的优先级不同的环境下显示。
真实内容尺寸 VS 合适尺寸
真实内容尺寸在Autolayout中扮演着输入角色。当一个视图存在真实内容尺寸是,系统会生成对应约束来表示尺寸,并且该约束被用于布局的计算。
另一方面,合适尺寸,相当于Autolayout发动机的输出。它是一个视图基于约束计算而得出的尺寸。如果一个视图使用Autolayout来布局子视图,系统便会基于其内容(也就是子视图的布局)计算出该视图的合适尺寸。
UIStackView就是一个好例子。排除任何其他约束,系统会基于其内容和属性计算出stack view的尺寸。从各种角度来看,stack view就好像是存在真实内容尺寸一样:你可以仅仅通过一个单独的垂直约束和一个水平约束就可以定义其位置,进而生成一个完整布局。但是它的尺寸是通过Autolayout进行计算的—-所以尺寸并不是作为Autolayout的输入。设置stack view的CHCR优先级也没有作用,因为实际上stack view并不存在真实内容尺寸。
如果你想要让stack view与外部其他视图(不是stack view的子视图)进行自适应尺寸适配,要么就创建明确的约束来捕获那些关系,要么就参照外部其他视图来修改stack view子视图的CHCR优先级。