UIViewController containment

iOS 5之前viewcontroller容器是不推荐使用的,这个时期的viewcontroller的任务是管理一个充满屏幕的view。从ios5 之后viewcontroller的任务即演变为管理自己的view。那么究竟是什么原因导致不可以将view controller的view当做子view来用呢

UIViewController containment_第1张图片
子viewcontroller的view作为子view

众所周知,UIWindow是app的根view,即是屏幕旋转即layout初始化信息源起的地方。由上图所知,子view controller的view是被插入rootviewcontroller的view层级中的,所以layout信息是不会触达这个子view controller的(比如 viewWillAppear)

ios5之前的自定义viewcontroller需要维护子controller的引用,并在自己收到各view事件之后手动触发各子controller的view事件方法, iOS 5之后不需要这样做,但各子view controller的viewwillappear等方法的调用时机规则可能跟独立的Viewcontroller不太一样。

UIViewController containment_第2张图片
寻找当前地点对应的地球另一面的地点

移动拿着铲子的人,地图会告诉你地球另一面的地点是哪里,点击雷达按钮,地图会轻弹并显示地点的名字。

上图中有两个map view controller,他们每个都可以单独处理拖拽,注释和更新地图。轻弹它们会出现两个新的viewcontroller,这些view controller会呈现位置的详细文字描述。且所有这些都包含在一个父viewcontroller中,持有子view controller的View并确保布局转屏运行正常。

UIViewController containment_第3张图片
viewcontroller代码

_startMapViewController作为子controller被添加,这会导致子controller的willMoveToParentViewController方法被调用,其view被添加进第一个容器view中,并显示通知其当前拥有parent controller(调用didMoveToParentViewController),_startGeoViewController初始化但并未添加进view层级

viewcontroller之间的切换

基于细粒度的viewcontroller包含api,使得创建及动画的效果可以应用在viewcontroller包含的场景中。切换两个controller的view的方法如下:(transitionFromViewController:toViewController:(...))

UIViewController containment_第4张图片
切换controller

在添加 toController为子controller时,先通知fromController它将被移除,如果它是包含view层级中的一员,则会调用其viewWillDisappear方法

toController会被通知其新的parentcontroller,并调用相关view事件方法。接着移除fromController

这个transition API方便地将旧view切换出去并为新view腾出空间,然而如果你想自定义切换效果,并只希望同一时刻只出现一个view,则需要对旧View调用 removefromsuperview,并对新view调用 addSubView。如果方法调用顺序出错会得到UIViewControllerHierarchyInconsistency的警告,比如在添加view之前即调用didMoveToParentViewController

为了使用UIViewAnimationOptionTransitionFlipFromTop动画,需要将子view添加为两个容器view的子view,而不是根controller的view,否则动画会是整个root view的效果。

父子controller之间的交互

它们之间只能通过暴露的api进行交互,且父controller需要负责layout子controller的view。父controller不应当修改子controller的view

custom container view controllers

任何时候子controller的view显示在容器的view层次中的时候,容器controller都需要与子controller建立联系并保证将所有适当的controller事件发送给子controller。

UIViewController containment_第5张图片
容器controller的view层次包含其他controller的view

容器制定规则,子controller需要遵守;容器决定子controller的内容何时出现在其view层次上,同时决定了其出现的位置和大小。必要情况下,容器controller也可以暴露属性和方法以供外界操作。

以navigationcontroller为例 ,当controller从栈上弹出或者入栈时,可以做动态切换

需要注意的是,shouldAutomaticallyForwardAppearanceMethods属性用来标识是否将容器controller的appearance相关的回调,直接传递给子controller。如果是NO,则容器controller负责告诉子controller何时view appear/disappear(通过调用子controller的beginAppearanceTransition:animated: 和 endAppearanceTransition方法来实现),不要直接调用子controller的view will/did appear/disappear方法。

在准备将controller添加进parentcontroller或者移除时,需要在动作之前操作willmovetoparentviewcontroller,之后操作didmovetocontroller,addchildviewcontroller自动调用willmoveto,didmoveto需要在出现的transition结束之后(或者如果没有transition则马上)马上调用, removefromparentviewcontroller自动调用didmovetocontroller,willmoveto需要在remove之前调用。

你可能感兴趣的:(UIViewController containment)