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

大家都知道,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
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 14,651评论 4 61
  • 1.自定义控件 a.继承某个控件 b.重写initWithFrame方法可以设置一些它的属性 c.在layouts...
    圍繞的城阅读 8,971评论 2 4
  • 我是个性格比较急躁的人,如果说是我正在忙其他事情,你问我这个事情怎么弄。我一般就会特别烦,所以我的脾气不是特别好的...
    123小屁孩闯天下阅读 2,523评论 0 0
  • 才刚刚离开家,在火车上过了一夜,现在就如此的想家,甚至鼻子都酸了。一直觉得自己小时候是很感性的,动不动就哭,后来,...
    城南少东家阅读 1,702评论 0 1

友情链接更多精彩内容