iOS学习系统push和popToRoot的特性

前提:有时候我们发现设置hidesBottomBarWhenPushedtrue无效.找到原因发现是因为如下代码:

if self.viewControllers.count == 1 {

            viewController.hidesBottomBarWhenPushed = **true**

} else {

            viewController.hidesBottomBarWhenPushed = **false**

}

现象:push第二个控制器的同时,将第一个push进来的控制器销毁时,再次push,Tabbar会显示出来

分析:

1.在自己的push的栈组中从栈顶到栈底遍历hidesBottomBarWhenPushed的值.如果期间有hidesBottomBarWhenPushed为true.则隐藏.没有为true的则显示

2.基于以上代码:如果需要把第一个push进来的控制器销毁.剩余的全部为false.导致tabbar出现

结论:

系统是从栈顶遍历到栈底.若hidesBottomBarWhenPushed有true.则隐藏tabbar.全部为false则显示

以下是系统UINavigationController的机制的相关打印:

image.png

即当前所有push的栈控制器.从栈顶找到栈底的hidesBottomBarWhenPushed,如果期间有一个为true.则隐藏.所有的均为false则显示.

****最后回归到问题:为什么push多次,在14.0系统上调用popToRootViewController时会隐藏Tabbar:****

先看结论:

 例如初始push信息如下:
 
 [
    3.ccc  
    
    2.bbb  
    
    1.aaa  

    0.rootVc
]

        
14.0以前,所以直接获取第一个的hidesBottomBarWhenPushed的值

[

    全部移除.只留了0(rootVc).

]



14.0版本以后,打印只有[3(ccc),0(rootVc)]

[

    3.ccc   **true**->**return** **true**->.隐藏   false往下找   可能14.0版本的popToRoot保留了[3,0],如果3为true,则直接隐藏.如果3为false.则继续往下找到0为fase显示

    [移除:

        2.bbb   **true** .隐藏   false往下找

        1.aaa   **true** .隐藏   **false**.显示

    ]

    0.rootVc **true**.隐藏  **false**.显示

]



14.0版本时,没有更改之前的逻辑.直接从前往后找为3的.

14.0以后的又正常了.更改了查找的逻辑.从[3,0],找最后一个.所以也正常了

分析打印:

 (14.0系统):调用之前的top,调用之后栈顶的viewControllers,和调用之前后的top

 大可-------------------------->调用之前 Optional(<TestHiddenBottomVC.DAKEViewController: 0x133e3d990>) Optional(**true**) 9

 大可--------------------->调用push以后 Optional([<TestHiddenBottomVC.DAKEViewController: 0x133e3d990>, <TestHiddenBottomVC.ViewController: 0x135008820>]) Optional(<TestHiddenBottomVC.ViewController: 0x135008820>)



 (14.5系统):

 大可-------------------------->调用之前 Optional(<TestHiddenBottomVC.DAKEViewController: 0x102624eb0>) Optional(**true**) 6

 大可--------------------->调用push以后 Optional([<TestHiddenBottomVC.DAKEViewController: 0x102624eb0>, <TestHiddenBottomVC.ViewController: 0x1026115e0>]) Optional(<TestHiddenBottomVC.ViewController: 0x1026115e0>)



(12.5系统):调用之前的top

大可-------------------------->AAAAPop Optional(<TestHiddenBottomVC.DAKEViewController: 0x100953de0>) Optional(**true**) 7

调用之后的栈顶控制器的navigationController.viewControllers,和调用之前后的top(因为该控制器已被销毁)

大可--------------------->dis 5 **nil** **nil**

调用之后的栈底控制器的navigationController.viewControllers,和调用之前后的top

大可--------------------->dis 0 Optional([<TestHiddenBottomVC.ViewController: 0x100827e40>]) Optional(<TestHiddenBottomVC.ViewController: 0x100827e40>)

分析描述:

 可以看出,在系统调用popToRootViewController过程中:

 14.0之前,保留栈底.使用栈底的hidesBottomBarWhenPushed的值

 14.0版本时.保留了[栈顶,栈底],从栈顶找到栈底,如果栈顶hidesBottomBarWhenPushed为ture,则隐藏,如果为false.则继续找栈底的hidesBottomBarWhenPushed.为true隐藏.为false则显示,

 14.5版本以后,保留了[栈顶,栈底],直接使用栈底的hidesBottomBarWhenPushed值,即rootVC的,true则隐藏,**false**.则显示,

 所以14.0系统版本的会隐藏

解决方案:

重写系统的popToRootViewControllerAnimated方法.设置:self.topViewController.hidesBottomBarWhenPushed = false即可

override func popToRootViewController(animated: Bool) -> [UIViewController]? {

/*14.0版本的会出现这种情况,14.5版本的没有这种情况,以下代码验证可以解决问题*/

        **if** (**self**.viewControllers.count > 1) {

            **let** vc = **self**.topViewController

            vc?.hidesBottomBarWhenPushed = **false**

        }

        **return** **super**.popToRootViewController(animated: animated)

}

根据这个特质.大家如果有有TabBar显隐相关的处理.可以参考一下

推荐阅读更多精彩内容