何时调用 addChildViewController
Posted
技术标签:
【中文标题】何时调用 addChildViewController【英文标题】:When to call addChildViewController 【发布时间】:2012-12-07 17:56:28 【问题描述】:我正在尝试使用类似于 UINavigationController 的东西,以便我可以自定义动画。首先,我只是使用 Apple 库存动画。这是我的 containerViewController:
- (void)loadView
// Set up content view
CGRect frame = [[UIScreen mainScreen] bounds];
_containerView = [[UIView alloc] initWithFrame:frame];
_containerView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight;
self.view = _containerView;
- (id)initWithInitialViewController:(UIViewController *)vc
self = [super init];
if (self)
_currentViewController = vc;
[self addChildViewController:_currentViewController];
[self.view addSubview:_currentViewController.view];
[self didMoveToParentViewController:self];
_subViewControllers = [[NSMutableArray alloc] initWithObjects:_currentViewController, nil];
return self;
- (void)pushChildViewController:(UIViewController *)vc animation:(UIViewAnimationOptions)animation
vc.view.frame = _containerView.frame;
[self addChildViewController:vc];
[self transitionFromViewController:_currentViewController toViewController:vc duration:0.3 options:animation animations:^
completion:^(BOOL finished)
[self.view addSubview:vc.view];
[vc didMoveToParentViewController:self];
[self.subViewControllers addObject:vc];
];
- (void)popChildViewController:(UIViewController *)vc WithAnimation:(UIViewAnimationOptions)animation
// Check that there is a view controller to pop to
if ([self.subViewControllers count] <= 0)
return;
NSInteger idx = [self.subViewControllers count] - 1;
UIViewController *toViewController = [_subViewControllers objectAtIndex:idx];
[vc willMoveToParentViewController:nil];
[self transitionFromViewController:vc toViewController:toViewController duration:0.3 options:animation animations:^
completion:^(BOOL finished)
[vc.view removeFromSuperview];
[vc removeFromParentViewController];
[self didMoveToParentViewController:toViewController];
[self.subViewControllers removeObjectAtIndex:idx];
];
我将这个 ContainerViewcontroller 作为窗口的 rootViewController。我可以添加我的初始 viewController 并推送一个视图控制器。当我尝试弹出时,我得到了
ContainerViewController[65240:c07] Unbalanced calls to begin/end appearance transitions for <SecondViewController: 0x8072130>.
我想知道我做错了什么。我想我的 initialViewController 仍然在 secondViewController 下面。有什么想法吗?谢谢!
【问题讨论】:
【参考方案1】:我不知道这是否是导致您的问题的原因,但不应该这样:
[self didMoveToParentViewController:toViewController];
是:
[toViewController didMoveToParentViewController:self];
另外,我不确定你在用 subViewControllers 数组做什么。这似乎是作为 UIViewController 属性的 childViewControllers 数组的重复。
另一件事我不确定是否正确。在您的 pop 方法中,您的 toViewController 是 _subViewControllers 数组中的最后一个控制器。你不希望它是倒数第二个吗?最后一个不应该是你要弹出的那个吗?您正在弹出 vc,这是您要传递给方法的控制器,我不明白。
这就是我制作类似控制器的导航的方式。在其包含行为中,它就像一个导航控制器,但没有导航栏,并允许不同的过渡动画:
@implementation ViewController
-(id)initWithRootViewController:(UIViewController *) rootVC
if (self = [super init])
[self addChildViewController:rootVC];
rootVC.view.frame = self.view.bounds;
[self.view addSubview:rootVC.view];
return self;
-(void)pushViewController:(UIViewController *) vc animation:(UIViewAnimationOptions)animation
vc.view.frame = self.view.bounds;
[self addChildViewController:vc];
[self transitionFromViewController:self.childViewControllers[self.childViewControllers.count -2] toViewController:vc duration:1 options:animation animations:nil
completion:^(BOOL finished)
[vc didMoveToParentViewController:self];
NSLog(@"%@",self.childViewControllers);
];
-(void)popViewControllerAnimation:(UIViewAnimationOptions)animation
[self transitionFromViewController:self.childViewControllers.lastObject toViewController:self.childViewControllers[self.childViewControllers.count -2] duration:1 options:animation animations:nil
completion:^(BOOL finished)
[self.childViewControllers.lastObject removeFromParentViewController];
NSLog(@"%@",self.childViewControllers);
];
-(void)popToRootControllerAnimation:(UIViewAnimationOptions)animation
[self transitionFromViewController:self.childViewControllers.lastObject toViewController:self.childViewControllers[0] duration:1 options:animation animations:nil
completion:^(BOOL finished)
for (int i = self.childViewControllers.count -1; i>0; i--)
[self.childViewControllers[i] removeFromParentViewController];
NSLog(@"%@",self.childViewControllers);
];
编辑后:通过向 IB 中的所有控制器(包括自定义容器控制器)添加导航栏,我能够使用此控制器复制后退按钮功能。我为任何将被推送的控制器添加了一个条形按钮,并将它们的标题设置为 nil(如果我将标题保留为“项目”,则会出现一些故障)。删除该标题会使按钮消失(在 IB 中),但您仍然可以在场景列表中与它建立连接。我向它添加了一个 IBOutlet,并添加了这段代码以获得我想要的功能:
- (void)viewWillAppear:(BOOL)animated
[super viewWillAppear:animated];
if (self.isMovingToParentViewController)
self.backButton.title = [self.parentViewController.childViewControllers[self.parentViewController.childViewControllers.count -2] navigationItem].title;
else
self.backButton.title = [self.parentViewController.childViewControllers[self.parentViewController.childViewControllers.count -3] title];
我已经展示了访问标题的两种不同方法——在 IB 中,您可以为我在 else 子句中使用的控制器设置标题,或者您可以像在 if 部分中那样使用 navigationItem 标题的条款。 else 子句中的“-3”是必要的,因为在调用 viewWillAppear 时,正在弹出的控制器仍在 childViewControllers 数组中。
【讨论】:
后续问题,我注意到当您使用 transitionFromViewController 时:您从看起来像 ParentViewController 或子视图控制器的容器中调用它。那是对的吗?如果是这样,为什么?所有的控制器都应该以这种方式从父级转换吗? @Crystal,是的,在这种情况下,我正在从父“人造导航”控制器进行所有转换。这与使用真正的导航控制器的工作方式相同,您可以从其中一个孩子调用 self.navigationController pushViewController:animated:。唯一的区别是这里你使用 self.parentViewController push ... @Crystal,您不必在父级中使用该转换代码,我只是认为它在这种情况下效果最好。我已经将该代码放在孩子中 - 然后您必须将该方法发送到 self.parentViewController。 当您实现自己的人造导航控制器或其他容器控制器时,您做了什么来访问navigationItem 属性或navigationController。我猜我不能再免费使用这些东西了…… @Crystal,我还没有尝试这样做,因为那不是我的重点。但是,不,我认为您不会免费获得。【参考方案2】:应该先调用addChildViewController
添加/删除可以参考这个大类,不用担心什么时候调用:
UIViewController + Container
- (void)containerAddChildViewController:(UIViewController *)childViewController
[self addChildViewController:childViewController];
[self.view addSubview:childViewController.view];
[childViewController didMoveToParentViewController:self];
- (void)containerRemoveChildViewController:(UIViewController *)childViewController
[childViewController willMoveToParentViewController:nil];
[childViewController.view removeFromSuperview];
[childViewController removeFromParentViewController];
【讨论】:
【参考方案3】:除了 rdelmar 的回答之外,您不应该调用 addView/removeFromSuperview transitionFromViewController
为您执行此操作,来自文档:
此方法将第二个视图控制器的视图添加到视图中 层次结构,然后执行动画中定义的动画 堵塞。动画完成后,它会删除第一个视图 视图层次结构中的控制器视图。
【讨论】:
谢谢我遇到了这个问题,我试图“改进”Apple 的实现,但没有查看 transitionFromViewController 文档......最后,transitionFromViewController 并没有真正为你做任何事情,好多了手动完成所有操作,只需使用 UIView 动画......我希望我能投票给你 +10以上是关于何时调用 addChildViewController的主要内容,如果未能解决你的问题,请参考以下文章
iOS开发 关于addChildViewController的理解
addChildViewController与viewWillAppearviewDidAppear关系说明
addChildViewController与viewWillAppearviewDidAppear关系说明
addChildViewController与viewWillAppearviewDidAppear关系说明