谈谈 iOS 中的 childViewController

使用场景

在 iOS 客户端中,多个 childViewController 的页面是个很常见的交互设计,目前已经广泛运用在各类的 APP 上,比较有代表性的类似网易新闻、今日头条这两个客户端。

代码实现

实现方式网上已经有很多了,这里就直接贴出代码,大概流程如下:

1
2
3
4
5
6
7
8
9
10
11
//添加一个 childViewController
UIViewController *vc = [UIViewController new];//子控制器
[self addChildViewController:vc];//添加到父控制器中
vc.view.frame = /*....*/;//设置 frame
[self.view addSubview:vc.view];//把子控制器的 view 添加到父控制器的 view 上面
[vc didMoveToParentViewController:self]; //子控制器被通知有了一个父控制器
//移除一个 childViewController
[vc willMoveToParentViewController:nil];//子控制器被通知即将解除父子关系
[vc.view removeFromSuperview];//把子控制器的 view 从到父控制器的 view 上面移除
[vc removeFromParentViewController];//真正的解除关系,会自己调用 [vc didMoveToParentViewController:nil]

实现上面部分,childViewController 的生命周期方法也就是 viewWillAppear、viewDidAppear等等这些,是不需要我们关心的,系统内部会自动帮我们调用。

手动管理 childViewController 的生命周期方法

有时候我们希望自己控制子控制器的生命周期方法,这里我们就需要一些额外的操作:

在 iOS 5 中,我们需要在父控制器中重写automaticallyForwardAppearanceAndRotationMethodsToChildViewControllers方法,并返回 NO;

在 iOS 6 及以后,需要重写 shouldAutomaticallyForwardAppearanceMethods方法,并返回 NO,这样系统就不会自动调用 childViewController 的生命周期了,全部交给我们自己处理。

不过我们需要注意的是,不能手动调用 viewWillAppear、viewDidAppear等等这些方法,而应该调用:

1
2
- (void)beginAppearanceTransition:(BOOL)isAppearing animated:(BOOL)animated;
- (void)endAppearanceTransition;

这两个方法来间接触发子控制器的生命周期,并且它们得成对使用:

isAppearing 设置为 YES : 触发 viewWillAppear: ;

isAppearing 设置为 NO : 触发 viewWillDisappear: ;

endAppearanceTransition 会触发 viewDidAppear: 以及 viewDidDisappear: 方法。

关于 childViewController 导致导航条穿透效果失效问题

所谓的导航条穿透效果一般是在用 UITableView 的时候,效果大概如下:

大概原理就是系统内部帮我们设置了 UITableView 的 contentInset 以及 contentOffset 属性,往下偏移了一定的高度,并且只有单个层级的情况下才能生效,所谓的单个层级可以理解为在一个 UIViewController 中 放一个 UITableView。一旦我们的界面 addChildViewController: ,就会失效,如果我们希望 childViewController 也能保持这种穿透效果,就需要拿到正确的 topLayoutGuide 以及 bottomLayoutGuide 值,然后设置 contentInsetcontentOffset 就行了:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
@implementation UIViewController (XXLayoutSupport)
- (id<UILayoutSupport>)xx_navigationBarTopLayoutGuide {
if (self.parentViewController &&
![self.parentViewController isKindOfClass:UINavigationController.class]) {
return self.parentViewController.xx_navigationBarTopLayoutGuide;
} else {
return self.topLayoutGuide;
}
}
- (id<UILayoutSupport>)xx_navigationBarBottomLayoutGuide {
if (self.parentViewController &&
![self.parentViewController isKindOfClass:UINavigationController.class]) {
return self.parentViewController.xx_navigationBarBottomLayoutGuide;
} else {
return self.bottomLayoutGuide;
}
}
@end
@implementation UIViewController (FixNavBarPenetrable)
- (void)xx_fixNavBarPenetrable {
if(!self.childViewControllers.count) return;
CGFloat statusBarHeight = [[UIApplication sharedApplication] statusBarFrame].size.height;
if (self.navigationController) {statusBarHeight = 0.0f;}
[self.childViewControllers enumerateObjectsUsingBlock:^(__kindof UIViewController * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
[obj.view.subviews enumerateObjectsUsingBlock:^(__kindof UIView * _Nonnull v_obj, NSUInteger v_idx, BOOL * _Nonnull v_stop) {
if ([v_obj isKindOfClass:[UIScrollView class]]) {
UIScrollView *tv = (UIScrollView *)v_obj;
const UIEdgeInsets insets = (obj.automaticallyAdjustsScrollViewInsets) ? UIEdgeInsetsMake(obj.xx_navigationBarTopLayoutGuide.length - statusBarHeight, 0.0f, obj.xx_navigationBarBottomLayoutGuide.length, 0.0f) : UIEdgeInsetsZero;
tv.contentInset = tv.scrollIndicatorInsets = insets;
tv.contentOffset = CGPointMake(insets.left, -insets.top);
*v_stop = YES;
}
}];
}];
}
@end