iOS - APP任意push新页面那些事

作者: 西蒙SIMON | 来源:发表于2017-02-28 00:17 被阅读1824次

    大家都知道,UINavigationController对象有一个方法pushViewController,用来做视图跳转,也是在iOS开发中常用的页面转换方法之一。大多数APP的结构一般都是,使用一个UITabBarController,每个tab上都是一个UINavigationController,然后各个tab上的navigationController自己管理各自的视图栈。

    Paste_Image.png

    这个时候,如果此时APP想要push一个新的VC,一般是在某个VC上调用:

    [self.navigationController pushViewController:newVC animated:YES];
    

    那问题来了,产品比较任性,如果有一个需求,就是要求做一个跳转工具,要求要能在任何地方(包括非VC中),都能在当前页面push新的VC,那又该怎么做呢?

    Paste_Image.png

    如果能拿到UITabBarController的对象,例如放在了delegate中(这里设想AppDelegateInstance就是delegate的对象,而UITabBarController的成员变量名为tabBar),也可以使用

    [AppDelegateInstance.tabBar.selectedViewController pushViewController:newVC animated:YES];
    

    好像能随时push新VC了......
    不对!状况又来了,如果在当前VC上可能又present出一个页面,即present出一个新的NAV,这时候:

    Paste_Image.png

    这个时候,有可能AppDelegateInstance.tabBar.selectedViewController就不是当前的NAV了,自然就不能正确push了。
    那怎么办呢?

    Paste_Image.png

    如果能拿到当前正在显示的NAV就好办多了,那样我直接拿这个NAV就可以push任何页面了。
    思路:
    1.当前显示的肯定有一个window包含了所有视图控制器、导航控制器;
    2.视图控制器、导航控制器出现的方式只有3中:tabBar selected、push、present;
    3.从“1”的window出发,找到每一层显示中的NAV或者VC,直到找到最后一层,就是当前显示的VC,就能拿到当前的NAV了。

    这里我创建了一个分类UIApplication+Visible,获取APP的主window,再递归进行当前显示中VC、NAV的查询:

    #import "UIApplication+Visible.h"
    
    @implementation UIApplication (Visible)
    
    - (UIWindow *)mainWindow {
        return self.delegate.window;
    }
    
    - (UIViewController *)visibleViewController {
        UIViewController *rootViewController = [self.mainWindow rootViewController];
        return [self getVisibleViewControllerFrom:rootViewController];
    }
    
    - (UIViewController *) getVisibleViewControllerFrom:(UIViewController *) vc {
        if ([vc isKindOfClass:[UINavigationController class]]) {
            return [self getVisibleViewControllerFrom:[((UINavigationController *) vc) visibleViewController]];
        } else if ([vc isKindOfClass:[UITabBarController class]]) {
            return [self getVisibleViewControllerFrom:[((UITabBarController *) vc) selectedViewController]];
        } else {
            if (vc.presentedViewController) {
                return [self getVisibleViewControllerFrom:vc.presentedViewController];
            } else {
                return vc;
            }
        }
        
    }
    
    - (UINavigationController *)visibleNavigationController {
        return [[self visibleViewController] navigationController];
    }
    @end
    

    然后你就这样调用

    UINavigationController *nav = [[UIApplication sharedApplication] visibleNavigationController];
    [nav pushViewController:newVC animated:YES];
    

    成功啦!!!

    Paste_Image.png

    最后,可能有些人要疑惑,为什么mainWindow方法中要使用delegate.window而不使用[UIApplication sharedApplication].keyWindow呢?
    因为keyWindow并不是一直都是APP的主界面的window,例如使用了UIAlertView,keyWindow就会变成alertView的window,这时候我们的轮子就不能跑了。

    好了,可能有些地方写得比较模糊、或者有缺陷,欢迎指出。

    Paste_Image.png

    相关文章

      网友评论

      • 请叫我喵_喵:不错,加油!
      • Lol刀妹:小伙子,可以
      • Ashen_:试试这个?
        func topviewController() -> UIViewController? {

        var aRoot = UIApplication.shared.keyWindow?.rootViewController

        while aRoot?.presentedViewController != nil {
        aRoot = aRoot?.presentedViewController

        if aRoot is UINavigationController {
        aRoot = (aRoot as? UINavigationController)?.visibleViewController
        } else if aRoot is UITabBarController {
        aRoot = (aRoot as? UITabBarController)?.selectedViewController
        }
        }

        return aRoot
        }
        西蒙SIMON:不能用keyWindow撒,而且如果rootViewController没有present其他VC的不就直接return了吗:joy:
      • akkkk47:把window考虑进去,还有很多情况。
        akkkk47:嗯 如果只有一个window当然没问题
        西蒙SIMON:是这么回事,我也只做过只有一个window的app:joy:
      • 开发全靠xib:貌似遍历响应者链也可以
      • 991aaa4f3361:假如当前显示的是一个alertviewcontroll你这个方法返回的就是一个alertvc了……
        我觉得你这种需求应该的写法是发个通知,然后在vc的基类里面接受通知,做跳转
        7ad0a4518b40:自定义模态不就行了,做成push的样子
        西蒙SIMON:如果使用的是keyWindow,返回的就是alertView的window,但是使用delegate的window,返回的就是app的主window。其实我这个需求主要是应对url router跳转的,要求app无论处在哪个界面都能够push/present出新的界面,而且要能置于最顶层,不能被覆盖住的。如果是VC基类接受通知,那么就会有很多个VC都收到这个通知了。
        Codepgq:@WDXWH 同意,如果哪里都可以push/present,比较乱。

      本文标题:iOS - APP任意push新页面那些事

      本文链接:https://www.haomeiwen.com/subject/anmtgttx.html