影响导航控制器中页面布局的几个属性

关于iOS的适配一直有许许多多的坑,今天来简单说一下最主要的几个影响布局的属性。

先来介绍几个主要属性

UIViewController 的 edgesForExtendedLayout
UIViewController 的 extendedLayoutIncludesOpaqueBars
UIViewController 的 automaticallyAdjustsScrollViewInsets
NavigationBar 的 translucent

  1. edgesForExtendedLayout属性的系统默认值为UIRectEdgeAll:意味着当导航控制器的导航栏为半透明效果时,子控制器self.view布局的起始位置将从屏幕边缘左上角开始。

  2. extendedLayoutIncludesOpaqueBars属性系统默认为NO,Opaque代表非透明,not Includes意味着导航栏不是半透明时,即便当前是UIRectEdgeAll,self.view的布局起始位置依旧是从导航栏下方开始。

  3. translucent属性值会决定导航栏是否有半透明效果。translucent为NO,意味着导航栏为非透明,此时如上文所述,即便当前是UIRectEdgeAll,由于extendedLayoutIncludesOpaqueBars为默认NO,self.view的布局起始位置依旧是从导航栏下方开始。

  4. 至于automaticallyAdjustsScrollViewInsets属性,系统默认值为YES。意味着当导航控制器的childViewController.view的上层视图为scollView类簇时,则系统会自动为该scrollView的contentInset(内边距属性)的Top值增加额外的64,于是内容就会下移64。

关于translucent属性需要特别注意的是:

根据官方文档所述:translucent会受navigationBar的backgroudImage属性的影响。也就是说当你使用了一张自定义图片作为navigationBar的背景图时,translucent的值将由系统根据该图片是否颜色值透明,来推断translucent是YES还是NO。

于是坑就来了:对于导航控制器中的各个childViewController,是共用同一个的navigationBar。当你在一个childViewController中自定义了navigationBar的背景图片,或是直接改变了translucent属性,此时再push或pop到另一个childViewController时,更改导航栏的半透明效果可能会影响到页面的布局起始位置,从而发生视图发生跳动,出现“意外”的上下偏移。

举个例子:从一个设置了导航栏不透明的控制器A,pop回到一个原本设置了导航栏透明的控制器B时,B页面发生了下移。

为避免该情况,应该将控制器B的extendedLayoutIncludesOpaqueBars设置为YES;或是当B页面viewWillAppear:时,再度将导航栏设置为半透明效果。

关于automaticallyAdjustsScrollViewInsets属性需要特别注意的是

以tableView为例,当你使用默认的创建方式,也就是UIRectEdgeAll+导航栏半透明的情况下,首行cell的位置将处于导航栏下方,也就是屏幕坐标系的(0, 64)位置处,此时上滑将会形成穿透效果导航栏的效果。

补充:当你的设计中出现导航栏穿透效果,自定义非透明的导航栏背景、导航栏完全透明等需求时,记得结合上述几点进行判断处理,避免发生预想之外的界面偏差。

文章目录
  1. 1. 先来介绍几个主要属性
  2. 2. 关于translucent属性需要特别注意的是:
  3. 3. 关于automaticallyAdjustsScrollViewInsets属性需要特别注意的是
|