iOS大咖说

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

2017-01-09  本文已影响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;
}

方法三:(推荐使用)

#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
上一篇下一篇

猜你喜欢

热点阅读