新版本iOS系统趋向于尽可能完全利用有限的屏幕以展示更多的用户数据,所以UINavigationBar和UITabBar默认都是半透明模糊效果,UIScrollView中的内容在它们后面也是隐约可见的。
在此情况下,系统对于作为controller view的第一个subView的全屏UIScrollView,会自动处理其contentInset,使其头部和尾部的内容起始和末尾时不会被UINavigationBar和UITabBar挡住。但是如果你关闭系统这一自动功能的话:
self.automaticallyAdjustsScrollViewInsets = NO;
UIScrollView的内容就会从frame的(0,0)位置开始显示,这样开头就被UINavigationBar,尾部就被UITabBar挡住而只能看到模糊一片,除非用户使劲下拉上拉。这时就需要你自己去设置contentInset了。
与此同时,由于现在view的frame都是从(0,0)开始,对于非UIScrollView及其子类就会有些麻烦了,因为如果子控件frame如果不考虑UINavigationBar的高度,就会被UINavigationBar挡住,当然我们可以自己在布局的时候加上UINavigationBar的高度就行了。
其实还有一种更方便的方法就是设置其edgesForExtendedLayout属性,该属性默认为UIRectEdgeAll,意为view会充分扩展至屏幕边缘包括上下左右,而不管有没有遮挡,此时就是view的frame即为整个屏幕。但是不像UIScrollView,其它view没有contentInset一说,所以automaticallyAdjustsScrollViewInsets对其并不起作用。此时,我们修改edgesForExtendedLayout属性:
self.edgesForExtendedLayout = UIRectEdgeNone;
系统就会自动根据UINavigationBar和statusBar将view下移64,frame从(0,64)开始。这样,我们在布局内部控件的时候依然可以从(0,0)开始,而不必担心上部被UINavigationBar遮挡了。
另一种情况就是我们不想要那种半透明模糊效果,设置UINavigationBar的translucent为NO。此时,view的frame也都会从(0,64)开始,而此时对于UIScrollView,automaticallyAdjustsScrollViewInsets就不再起作用了。
总结:UIScrollView如果要实现穿透UINavigationBar的效果,其frame就必须从(0,0)开始,然后设置合适的contentInset,如果UINavigationBar下方还有悬浮view的话,就不能依赖系统的automaticallyAdjustsScrollViewInsets了,需要自己来算contentInset。还有其他的特殊需求,就需要组合使用以上属性了。