美文网首页iOS开发实践iOS精选博文iOS超神之路
【8行代码教你搞定导航控制器全屏滑动返回效果】 |那些人追的干货

【8行代码教你搞定导航控制器全屏滑动返回效果】 |那些人追的干货

作者: 袁峥 | 来源:发表于2015-08-01 11:59 被阅读14880次

    前言

    此次文章,讲述的是导航控制器全屏滑动返回效果,而且代码量非常少,10行内搞定。
    效果如图:


    效果.gif

    一、自定义导航控制器

    目的:以后需要使用全屏滑动返回功能,就使用自己定义的导航控制器。

    二、分析导航控制器侧滑功能

    效果:导航控制器默认自带了侧滑功能,当用户在界面的左边滑动的时候,就会有侧滑功能。

    系统自带的侧滑效果:

    侧滑.gif

    分析:
    1.导航控制器的view自带了滑动手势,只不过手势的触发范围只能在左边。

    2.当用户在界面左边拖动,就会触发滑动手势方法,并且有滑动返回功能,说明系统手势触发的方法已经实现了滑动返回功能。

    3.为什么说系统手势触发的方法已经实现了滑动返回功能?

    原因:

    • 创建滑动手势对象的时候,需要绑定监听者,当触发手势的时候会调用target的action。
    UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:action];
    
    
    • 当用户在界面左边滑动,有滑动返回功能,这是因为触发手势了,调用target的action方法,说明action方法内部实现滑动返回功能,否则就不会有滑动返回效果。

    三、实现全屏滑动功能分析

    • 打印导航控制器自带的滑动手势,看下它的真实面目。
    • 系统自带的滑动手势interactivePopGestureRecognizer
    //  self指向的导航控制器,在导航控制器的viewDidLoad方法打印
    - (void)viewDidLoad {
        [super viewDidLoad];
        NSLog(@"%@",self.interactivePopGestureRecognizer);
    }
    

    打印结果图片:

    Snip20150801_1.png
    • 由图中可知:
    • 1.系统自带的手势是UIScreenEdgePanGestureRecognizer类型对象,屏幕边缘滑动手势
    • 3.系统自带手势target是_UINavigationInteractiveTransition类型的对象
    • 4.target调用的action方法名叫handleNavigationTransition:

    分析:

    • UIScreenEdgePanGestureRecognizer,看名称就知道,这个手势的范围只能在屏幕的周边,就是因为这个手势,系统自带的滑动效果,只能实现侧边滑动。

    四、如何实现全屏滑动功能

    • 给自己的导航控制器,添加一个全屏的滑动手势,调用系统自带滑动手势的targetaction方法,利用系统实现的滑动返回功能,加上自己全屏滑动手势,就有全屏滑动功能了。

    • 问题:如何拿到系统自带的target对象?,action方法名已经知道,而且系统肯定在target对象实现了,只要拿到target对象,调用这个方法就行。

    • 通过打印系统自带的滑动手势的代理,发现正好是_UINavigationInteractiveTransition对象,因此我猜测这个代理对象就是target对象,只要拿到它,就拿到系统自带滑动手势的target对象。

        // 打印系统自带滑动手势的代理对象
        NSLog(@"%@",self.interactivePopGestureRecognizer.delegate);
    
    
    
    • 打印图片:
    Snip20150801_2.png
    • 导航控制器全屏滑动注意点:

    • 1.禁止系统自带滑动手势使用

    • 2.只有导航控制器的非根控制器才需要触发手势,使用手势代理,控制手势触发。

    • 全屏滑动代码实现

    - (void)viewDidLoad {
        [super viewDidLoad];
    
        // 获取系统自带滑动手势的target对象
        id target = self.interactivePopGestureRecognizer.delegate;
    
        // 创建全屏滑动手势,调用系统自带滑动手势的target的action方法
        UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];
    
        // 设置手势代理,拦截手势触发
        pan.delegate = self;
    
        // 给导航控制器的view添加全屏滑动手势
        [self.view addGestureRecognizer:pan];
    
        // 禁止使用系统自带的滑动手势
        self.interactivePopGestureRecognizer.enabled = NO;
    
    }
    
    // 什么时候调用:每次触发手势之前都会询问下代理,是否触发。
    // 作用:拦截手势触发
    - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
    {
        // 注意:只有非根控制器才有滑动返回功能,根控制器没有。
        // 判断导航控制器是否只有一个子控制器,如果只有一个子控制器,肯定是根控制器
        if (self.childViewControllers.count == 1) {
            // 表示用户在根控制器界面,就不需要触发滑动手势,
            return NO;
        }
        return YES;
    }
    

    联系方式

    如果你喜欢这篇文章,可以继续关注我,微博:吖了个峥,欢迎交流。

    相关文章

      网友评论

      • 3dace6653964: 遇到bug, 此种方式导致tableView左滑删除无反应
      • 29f1dcdf0613:你这个写的太好了
      • oceanfive:如果系统的方法名称handleNavigationTransition:更改了话,应该会导致app崩溃了吧,这种情况怎么解决?
      • 九林:不错
      • e2f2d779c022:博主写的真好, 我自己封装了一个类似的框架,感兴趣的可以看下https://github.com/Chris-Pan/JPNavigationController
      • Nedoloroso:除了之前评论说的不能和table左滑删除共存之外,还发现另外一个问题,就是我现在需要在返回的时候做其他的操作,比如发个请求、发个通知什么的,重写了leftBatButton,滑动返回根本不调用
        川农鉴黄师:@Nedoloroso 楼主第二个问题实在那个固件版本出现的问题,iOS10只有第一种情况!
      • 诸葛俊伟:handleNavigationTransition: 这个方法在swift3中没有用啊。识别不了,怎么办?
      • SuDream:大哥,我用这个遇到个很严重的问题啊我不知道咋解决,就是我有一个页面要直接返回root试图.给了这个手势.怎么处理单独页面的,在线等.急,望大神解决
        SuDream:@峥吖 那就没啥办法解决这问题了?,有个需求就这样啊,有个页面需要直接返回 rootVC ,不行能把这玩意就这个页面给禁止了吗?
        袁峥:@SuDream 滑动只能回到上一个
      • 3767d1e0a5fb:有个 bug,当我要为 tableview 的 cell 提供左滑删除功能的时候,会有冲突。tableviewcell 的左滑没反应
        袁峥:@白小先生 恩 有冲突 ,暂不同时支持两个操作
      • 2a09405648f1:这个方案,使用过程中,发现问题。不管从那个方向滑动都会触发popViewController,导致调用了viewWillDisAppear和viewWillAppear。 具体的话,铮哥你在这两个地方打上断点,然后在屏幕上每个方向滑动试试
        若雨千寻:这个有什么好的解决办法吗?
      • dymost:不错
      • 507c8564b280:用这个跳转到第一个页面的时候,右上方有黑边?这是怎么了?
        sclcoder:@管城的waiting 不用这个方法也有这个问题的
      • 91阿生:这样做的话会与其他手势冲突.不知道怎么解决
      • d7e0d8007469:如果在导航控制器里的最后一个视图控制器中添加如下代码:
        id target = self.navigationController.interactivePopGestureRecognizer.delegate;

        UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];

        [self.view addGestureRecognizer:pan];

        导航控制器中除根控制器外的所有视图控制器都能实现侧滑,而且不需要只滑动边缘
      • 刘小壮:@啊崢 你这个程序有一个bug,我在非rootViewController的控制器中进行左滑操作的时候,并不是返回的右滑手势,滑动之后虽然界面上没有任何变化但是会导致当前控制器卡死。这个问题是重写的gestureRecognizerShouldBegin:方法不够严谨,应该加一个滑动手势方向的判断,是否返回的滑动方向,具体的原因我就不详细说了。
        袁峥:@刘小壮 没听明白,能在说下吗
      • 7ee94c706480:有个问题,,我写了,当导航控制器有多个子控制器时,手势无效,刚开始滑动几次还可以,但后面就不行了,,,连其他交互都不能用
      • keshiim:不好意思,亲测,确实好使,我刚才写代码的时候忘记设置代理了。这个是我写的demo地址https://github.com/keshiim/NavigationController-
      • keshiim:我自定义navigationcontroller试了下,不好用。还是我的方法不对
      • sclcoder:so easy
      • 袁峥:@从今以后 赞一个 分析的很透彻
        不知什么人:@啊崢 惭愧,我也只是搬运以前看到的博文中的一段而已.这个方法看过三篇博文,你这个最为简洁.
      • 叉叉歪大魔王:向左滑 滑到边界也会返回
      • c2aefd504819:有个bug ,就是滑动返回的时候UIBarButtonItem 有重叠的情况出现!
        d99f31dc0909:@c2aefd504819 你解决了吗
      • b8765e130bb0:有Demo 么?
      • shareMind:只能说,大神太厉害,看其他人写的代码,那个复杂,runtime换系统的方法都用上了,你这。。。也太简单了。就是不知道能不能上架?
      • 家丁三锅:确实很方便,但私有APi不能上架啊
        不知什么人:"不论是 kvc 还是 selector 反射,都是利用 objc runtime 完成的,而到了这一层,真的就没啥公有私有可言了。设想你就是开发 Apple 私有 API 检查工具的工程师,给你一个 ipa 的包,你会如何检查出其中有没有私有 API 呢?

        首先,这个检查一定是个静态检查吧,不可能是运行时检查,因为代码逻辑那么复杂,把程序跑起来看所有 objc_msgSend 中包不包括私有调用这件事太不现实了。
        对 ipa 文件做静态检查的话肯定是去分析 Mach-O 可执行文件,因为这时很多源代码级别的信息已经丢失,经分析可以采取下面几种手段:

        是否 link 了私有 framework 或者公开 framework 中的私有符号,这可以防止开发者把私有 header 都 dump 出来供程序直接调用。
        同上,使用@selector(_private_sel)加上-performSelector:的方式直接调用私有 API。
        扫描所有符号,查看是否有继承自私有类,重载私有方法,方法名是否有重合。
        扫描所有string,看字符串常量段是否出现和私有 API 对应的。

        我觉得前三条被 catch 住的可能性最高,也最容易被检查出来。再来看我们用到用字符串的方法 kvc 和 反射 selector,应该属于最后一条,这时候就很难抉择了,拿 handleNavigationTransition: 来说,看上去人畜无害啊,我自己类里面的方法也完全可能命名出这个来,所以单单凭借字符串命中私有 API 判定,苹果很容易误伤一大票开发者。
        综上,我觉得使用字符串的方式使用私有 API 是相对安全的,我们的 App 马上要提交审核,如果过了几天你还能读到这段文字,说明我的猜想是木有错的,大家可以放心使用。"
        审核通过了,可以看下这篇文章.http://blog.sunnyxx.com/2015/06/07/fullscreen-pop-gesture/
        家丁三锅:@碧野MAX handleNavigationTransition:
        6fdb0c58ceca:@Cosy_Freedom 没看到私有API啊
      • SilenceZhou:能不能发个Demo
      • SilenceZhou:怎么实现不了????
      • ea9ae456d0de:简单实用 :kissing_heart:
      • 十一岁的加重:还是阿峥厉害,多写点实用的东西吧
      • 南栀倾寒:这里有一篇类似的 可以参考
      • 南栀倾寒:刁刁叼 这个东西各种解决方案 还是看类的内部结构比较省劲

      本文标题:【8行代码教你搞定导航控制器全屏滑动返回效果】 |那些人追的干货

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