【问题标题】:When to call addChildViewController何时调用 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 下面。有什么想法吗?谢谢!

【问题讨论】:

    标签: ios


    【解决方案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
      猜你喜欢
      • 2015-11-27
      • 2011-12-26
      • 2013-08-18
      • 2012-04-25
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2012-01-17
      • 1970-01-01
      相关资源
      最近更新 更多