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

我们都知道,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
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容