何时调用 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关系说明

使用 addChildViewController 时的 IBAction 不起作用

AddChildViewController