iOS AutoLayout还可以这样玩
data:image/s3,"s3://crabby-images/ad327/ad3275630b5ee50a948a2dc10e1677e40a3a7f0c" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
网上关于Auto Layout
的文章繁多, 但是总觉得有些内容并没有说清楚。所以在这篇博文,我将通过一个demo演示Auto Layout
中几个容易被忽略,但是又相当重要的概念
demo效果
data:image/s3,"s3://crabby-images/16459/164596a95f072df49964dd6d0d726112a0b14c1d" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
demo 中要通过约束要实现的效果如下:
- backgroundView width>=60;
- backgroundView的width随着Label中内容的增加而增加
- backgroundView与其superView的margin>=30。也就是当Label的内容过长时,会显示出省略号。
概念
要实现demo 中的效果前,需要知道下面的几个概念:
1, Constraint equalities
(暂且叫做约束等式):这个就是常见的约束类型。不再赘述,举几个例子:
Red.top = 1.0 * Superview.top + 20.0 Superview.bottom = 1.0 * Red.bottom + 20.0 Red.top = 1.0 * Blue.top + 0.0 Red.bottom = 1.0 * Blue.bottom + 0.0
2, Constraint Inequalities
(暂且叫做约束不等式):它指定一个区域而不是一个确切的值。
// Setting the minimum width View.width >= 0.0 * NotAnAttribute + 40.0 // Setting the maximum width View.width <= 0.0 * NotAnAttribute + 280.0
3,Constraint Priorities
约束优先级 : 约束优级是一个0-1000的数值。当两个约束出现冲突时,优先级高的约束将保留,优先级低的约束失效。
4,Intrinsic Content Size
内在内容size: 有些view 可以根据其内容设定其size。比如说UILabel,UIButton等,他们的size刚好能够容纳其内容支持Intrinsic Content Size
的view如下:
data:image/s3,"s3://crabby-images/d1d4a/d1d4af5d059769ed0d3e3af5305b36cfd1d96ae2" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
5, content-hugging priorities
抗拉伸优先级(默认250) :这个优先级与 Intrinsic Content Size
相关。假如一个Label的Intrinsic Content Size
width=50, 现在添加一个width=60的约束(默认优先级是1000)现在Label就会拉伸 。如果将 约束width=60的优先级设置成小于250的值,Label就不会被拉伸。
6, compression-resistance priorities
抗压缩优先级(默认750):这个优先级也与 Intrinsic Content Size
相关。假如一个Label的Intrinsic Content Size
width=50, 现在添加一个width=40的约束(默认优先级是1000)现在Label就会压缩 。如果将 约束width=40的优先级设置成小于750的值,Label就不会被压缩。 可以看出 content-hugging priorities
和compression-resistance priorities
用于抵抗其他约束对viewIntrinsic Content Size
的改变。
###实战介绍完上面的概念,让我们使用他们完成上面demo效果。
1, 新建工程,打开storyboard。添加backgroundView(继承自UIView), 设置背景颜色为灰色并添加 水平居中和垂直居中约束。此时会出现约束错误,这是因为这个backgroundView只设置了position,没有设置size。 不必担心,一会就可解决这个问题;
约束如下:
data:image/s3,"s3://crabby-images/ce2b8/ce2b82b48dda171c25a7f4ed201ed28d58c04d64" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
data:image/s3,"s3://crabby-images/5f1b5/5f1b595a2dcace639c151f5694cbb6ba8dea3a19" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
view层级关系如下:
data:image/s3,"s3://crabby-images/9e815/9e815f1f353547b391ac6e3bf8d6ebaae328f455" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
添加下列约束:
data:image/s3,"s3://crabby-images/b5f1c/b5f1cd7a5e16258a774a5583459d2130730d0592" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
data:image/s3,"s3://crabby-images/fc2d0/fc2d0a706c26a3fa5e594f21c41ab37ff0158784" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
此时的约束没有任何错误了显示效果如下:
data:image/s3,"s3://crabby-images/d4588/d45880c20962c3e80374945b4a62d155df01f5c2" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
3, 此时通过给Label添加文字,backgroundView也会相应增大(只分析水平方向)。为什么增大? 这是因为Label的text决定Intrinsic Content Size
的大小, text增多,Intrinsic Content Size
相应增加。Label与backgroundView 的margin为10。所以根据这一系列的关系导致了 给Label添加文字,backgroundView也会相应增大。
data:image/s3,"s3://crabby-images/eece9/eece99f891b7a339e99cbdf1fb19eb642474b603" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
4, 现在有个问题, 当label中的文字太多时,backgroundView的宽度会超出其父view范围,这显然不太好。添加一个约束使backgroundView与其父view之间有一个最小的margin。 这时就要使用约束不等式了。约束不等式可以指定一个范围而不是一个确切的数值。看下面的例子。
data:image/s3,"s3://crabby-images/6dde9/6dde97b31c841dd36e8045514bfe3f1686ad661a" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
添加如下约束, 这样backgroundView和superView 之间的margin最小是30。当label中的内容过长时,内容就会被压缩。But Why?
data:image/s3,"s3://crabby-images/b5d57/b5d57726b76e23aacf43f101fbaf906430639fe1" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
data:image/s3,"s3://crabby-images/21fb6/21fb69a3f72e09447fea2d20ad94c76c56daa682" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
5 , backgroundView 的width随Label中内容的增加而增加。 当backgroundView 与其superView 的margin=30后,再增加 backgroundView 的width就会产生约束冲突了(与之前设置的 ‘backgroundView和superView 之间的margin最小是30’ 这条约束产生冲突 )。那么stroyboard是如何解决冲突的那? 对,就是优先级。‘backgroundView和superView 之间的margin最小是30’这条约束的优先级时1000, 是最大优先级。 Label 的抗压缩优先级默认750,1000>750, Label这能被压缩了(内容省略了一部分)。如果你将这两个优先级大小交换一下,backgroundView与superView 就之间不会出现margin了。
6, 当label中有一个字母是, 效果是这样的。 好丑。。。。
data:image/s3,"s3://crabby-images/d68bb/d68bbd8be0920e6fdaacca18f11d6070e8105cfb" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
我们给backgroundView设置一个最小width约束。 我们设置backgroundView的width =60约束, 并设置优先级996。(当然,你也可以使用width >=60这样的约束。这里为了演示优先级)现在的效果是这个样子的:
data:image/s3,"s3://crabby-images/82235/82235e979e5c790c7e70dec7f22d80c8050252c1" alt="iOS AutoLayout还可以这样玩 iOS AutoLayout还可以这样玩"
又出问题了。。。 虽然backgroundView width=60确实起作用了,但是Label内容增加时backgroundView的width并没有增加。这还是优先级的问题。Label 的抗压缩优先级是750, backgroundView width=60的优先级时996,996>750, 当然会出问题了。 现在只需让 Label的抗压缩优先级大于backgroundView width=60的优先级就可以了 。
7, 关于抗拉伸优先级,没想到很好的例子,就不演示了。
结语
一种布局效果能通过多种约束方式实现出来,关键是思路清晰,一步一步的来。参考资料,当然是 官网了。希望对你有帮助,如有帮助就给个喜欢吧。