对于iOS开发者来说,最早熟悉的布局应该是使用frame布局了,就是我们平常使用的frame,size等对控件进行布局,但平常也仅限于使用,很长时间以来从未对布局进行一个系统的认识和区分,有的时候可能会导致疑惑。希望可以通过这次梳理,把layout的内容理解更透彻。下面废话不多说,让我们来认识一下从iOS的布局方式。
在这之前先熟悉几个API,这几个也是我们经常拿过来比较的。
- (CGSize)sizeThatFits:(CGSize)size
- (void)sizeToFit
- (void)layoutSubviews
- (void)layoutIfNeeded
- (void)setNeedsLayout
- (void)drawRect:(CGRect)rect
在layoutSubviews这个方法中会完成最终布局,那么什么时候会调用它呢?
苹果官方文档给出了答案,layoutSubviews在以下情况下会被调用:
1、init初始化不会触发layoutSubviews但是是用initWithFrame 进行初始化时,当rect的值不为CGRectZero时,会触发。
[[UIView alloc]initWithFrame:CGRectZero]
不会触发layoutSubviews
2、addSubview会触发layoutSubviews
3、设置view的Frame会触发layoutSubviews,当然前提是frame的值设置前后发生了变化
准确的说应该是view的size发生了变化,如果仅仅改变view的origin并不会触发,这里说的设置frame应该说的是使用init(并非使用initWithFrame)初始化方法后的首次设置frame,注意 frame不能为CGRectZero。
4、滚动一个UIScrollView会触发layoutSubviews
5、旋转Screen(包括手动调用view的transform属性旋转)会触发父UIView上的layoutSubviews事件
6、改变一个UIView大小的时候也会触发父UIView上的
注意:
1.以上这些触发事件顺序是父视图先进行layoutSubviews
完成之后,子视图再调用layoutSubviews
,在视图第一次显示之前会统一调用一次layoutSubviews
在此之前addSubview、改变size、transForm旋转等都不会触发layoutSubviews
。
2.layoutSubviews
这个方法,默认系统自动调用,如果想在这个方法里布局需要重写
(不要忘记[super layoutSubviews]
)
3.更新布局调用setNeedsLayout
方法, 它会标记为需要重新布局,异步调用layoutIfNeeded刷新布局,不会立即刷新,但layoutSubviews
一定会被调用
layoutIfNeeded
如果,有需要刷新的标记,立即调用layoutSubviews
进行布局(如果没有标记,不会调用layoutSubviews
)
4.如果要立即刷新,要先调用[view setNeedsLayout]
,把标记设为需要布局,然后马上调用[view layoutIfNeeded]
,实现布局,在视图第一次显示之前,标记总是“需要刷新”的,可以直接调用[view layoutIfNeeded]
。
关于重绘:
drawRect:(CGRect)rect
在该方法进行视图绘制,此方法一般是系统调用。若想手动调用执行重绘任务使用setNeedsDisplay
方法(标记为需要重绘,异步调用drawRect
),局部绘制调用setNeedsDisplayInRect:(CGRect)invalidRect
方法(标记为需要局部重绘)。
关于sizeToFit
sizeToFit
会自动调用sizeThatFits
方法;
sizeToFit
不应该在子类中被重写,应该重写sizeThatFits
sizeThatFits
传入的参数是receiver当前的size,返回一个适合的size
sizeToFit
可以被手动直接调用
sizeToFit
和sizeThatFits
方法都没有递归,对subviews也不负责,只负责自己
layoutSubviews系统执行过程
layoutSubviews
方法调用先于drawRect
,setNeedsLayout
在receiver标上一个需要被重新布局的标记,在系统runloop的下一个周期自动调用layoutSubviews
layoutIfNeeded
方法如其名,UIKit会判断该receiver是否需要layout.根据Apple官方文档,layoutIfNeeded
方法应该是这样的
layoutIfNeeded
遍历的不是superview链,应该是subviews链
drawRect
是对receiver的重绘,能获得context
setNeedDisplay
在receiver标上一个需要被重新绘图的标记,在下一个draw周期自动重绘,iPhone的刷新频率是60hz,也就是1/60秒后重绘
关于drawRect:
在视图第一次显示之前会调用一次,除非手动用setNeedDisplay
或者UIView的contentMode属性设置为 UIViewContentModeRedraw,否则任意改变布局都不会重新绘制。