美文网首页iOS大咖说
知识点总结:04-iOS自定义导航控制器侧滑返回的多种实现方法

知识点总结:04-iOS自定义导航控制器侧滑返回的多种实现方法

作者: 枫之叶_小乙哥 | 来源:发表于2017-01-09 23:03 被阅读36次

    我们都知道,iOS7导航控制器默认自带了侧滑功能,当用户在界面的左边滑动的时候,就会有侧滑功能。但是如果我们从从导航控制器的返回按钮,就发现系统所带的侧滑返回功能无法使用。因此为了解决此问题,有以下方法实现:

    方法一:导航控制器全屏滑动返回效果
    当用户在界面左边拖动,就会触发滑动手势方法,并且有滑动返回功能,说明系统手势触发了方法,即调用了target的action方法,也就是说action方法内已经实现侧滑返回。 系统自带的滑动手势interactivePopGestureRecognizer

    // self指向的导航控制器,在导航控制器的viewDidLoad方法打印
    - (void)viewDidLoad { 
          [super viewDidLoad]; 
          NSLog(@"%@",self.interactivePopGestureRecognizer);
    }
    

    打印结果知:
    1.系统自带的手势是UIScreenEdgePanGestureRecognizer类型对象,屏幕边缘滑动手势
    2.系统自带手势target是_UINavigationInteractiveTransition类型的对象
    3.target调用的action方法名叫handleNavigationTransition:

    全屏滑动代码块实现

    - (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;
    }
    

    导航控制器全屏滑动注意点:
    - 1.禁止系统自带滑动手势使用。
    - 2.只有导航控制器的非根控制器才需要触发手势,使用手势代理,控制手势触发。

      完成、这样就搞定了, 亲测此方法会与界面有滑动手势的产生冲突,因此有第二种方法!
    

    以上方法参考原文链接如下:
    http://www.cocoachina.com/ios/20150811/12897.html?utm_source=tuicool)

    方法二:实现自定义导航控制器边缘滑动返回
    方法二的实现原理和方法一一样,只不过它还是使用的系统的边缘手势实现侧滑返回功能。只需要在每个类里面添加如下代码块:
    写在.m中, 别忘了遵守 UIGestureRecognizerDelegate协议。

    //当前导航控制器@property (nonatomic, strong) UIViewController *currentShowVC;
    
    -(void)viewWillAppear:(BOOL)animated {
    //设置代理
    self.navigationController.interactivePopGestureRecognizer.delegate =(id)self;
    
    //启用系统自带的滑动手势 
    self.navigationController.interactivePopGestureRecognizer.enabled = YES;
    
    //判断导航控制器是否只有一个子控制器,如果只有一个子控制器,肯定是根控制器,这里我的项目是有tabbar,所以在页面切换之间设置是否显示tababr。 
        if (self.navigationController.viewControllers.count == 1){ 
      //将当前导航控制器置空
             self.currentShowVC = Nil;
             [SharedAppDelegate setTabBarHidden:NO animated:YES];   
        }else{ 如果不是根控制器,就设置当前导航控制器为其本身。     
             self.currentShowVC = self; 
              [SharedAppDelegate setTabBarHidden:YES animated:NO]; 
    }
    }
    

    手势的代理方法

    -(BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer{ 
          if (gestureRecognizer ==     
              self.navigationController.interactivePopGestureRecognizer) { //当前导航控制器是根视图控制器
               //the most important 
              return (self.currentShowVC ==         
              self.navigationController.topViewController); 
    // 不要隐藏tabbar 
            [SharedAppDelegate setTabBarHidden:NO animated:NO]; 
    }
        return YES;
    }
    

    方法三:(推荐使用)

    • 自定义UINavigationController
    #import "ZGKNavigationViewController.h"
    
    @interface ZGKNavigationViewController ()<UIGestureRecognizerDelegate>
    
    @end
    
    @implementation ZGKNavigationViewController
    
    - (void)viewDidLoad {
        [super viewDidLoad];
    
        // self.interactivePopGestureRecognizer.delegate = nil;就是什么时候都可以滑动手势,会有bug,无法点击settingBtn
        // 设置手势代理
        self.interactivePopGestureRecognizer.delegate = self;
        
        //    self.interactivePopGestureRecognizer.enabled = YES;
        
        // 设置导航控制器navigationBar和tabBar的背景颜色可以解决导航栏黑色点的bug
        [self.navigationBar setBackgroundImage:[UIImage imageNamed:@"navigationbarBackgroundWhite"] forBarMetrics:UIBarMetricsDefault];
    }
    
    - (void)didReceiveMemoryWarning {
        [super didReceiveMemoryWarning];
        // Dispose of any resources that can be recreated.
    }
    
    
    /**
     *  重写push方法的目的 : 拦截所有push进来的子控制器(包括代码和storyBoard实现的)
     *
     *  @param viewController 刚刚push进来的子控制器
     */
    - (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated{
        if (self.childViewControllers.count > 0) { // 如果viewController不是最早push进来的子控制器,则不用设置返回按钮
            // 左上角
            UIButton *backButton = [UIButton buttonWithType:UIButtonTypeCustom];
            [backButton setImage:[UIImage imageNamed:@"navigationButtonReturn"] forState:UIControlStateNormal];
            [backButton setImage:[UIImage imageNamed:@"navigationButtonReturnClick"] forState:UIControlStateHighlighted];
            [backButton setTitle:@"返回" forState:UIControlStateNormal];
            [backButton setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
            [backButton setTitleColor:[UIColor redColor] forState:UIControlStateHighlighted];
            [backButton sizeToFit];
            // 这句代码放在sizeToFit后面
            backButton.contentEdgeInsets = UIEdgeInsetsMake(0, -20, 0, 0);
            [backButton addTarget:self action:@selector(back) forControlEvents:UIControlEventTouchUpInside];
            viewController.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:backButton];
            // 隐藏底部的工具条
            viewController.hidesBottomBarWhenPushed = YES;
        }
        
        // 所有设置搞定后, 再push控制器(不要忘记调用父类方法)
        [super pushViewController:viewController animated:animated];
    
    }
    
    - (void)back{
        [self popViewControllerAnimated:YES];
    }
    
    //- (UIViewController *)popViewControllerAnimated:(BOOL)animated
    //{
    //    return [super popViewControllerAnimated:NO];
    //}
    //
    //- (NSArray<UIViewController *> *)popToViewController:(UIViewController *)viewController animated:(BOOL)animated
    //{
    //    return [super popToViewController:viewController animated:NO];
    //}
    //
    //- (NSArray<UIViewController *> *)popToRootViewControllerAnimated:(BOOL)animated
    //{
    //    return [super popToRootViewControllerAnimated:NO];
    //}
    
    #pragma mark - <UIGestureRecognizerDelegate>
    /**
     *  手势识别器对象会调用这个代理方法来决定手势是否有效
     *
     *  @return YES : 手势有效, NO : 手势无效
     */
    - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
    {
    //    if (self.childViewControllers.count == 1) {
    //        return NO;
    //    }
    //    
    //    return YES;
        
        // 手势何时有效 : 当导航控制器的子控制器个数 > 1就有效
        return self.childViewControllers.count > 1;
    }
    
    @end
    

    相关文章

      网友评论

        本文标题:知识点总结:04-iOS自定义导航控制器侧滑返回的多种实现方法

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