iOS系统右滑返回全局控制方案

简介:

前言

今天有个小需求,在点击导航条上的返回按钮之前要调用某个API,并弹出UIAlertView来显示,根据用户的选项判断是否是返回还是继续留在当前控制器。举个简单的例子,当点击导航条上的左上角返回按钮时,就调用我们的API来提示是否知道,点击知道则返回,点击不知道则继续留在当前控制器。

那么问题来了,导航自带的右滑返回手势在点击系统的返回按钮时,不会没有办法处理,那是自动的,因此就要想办法改成leftBarButtonItem了,但是使用了leftBarButtonItem就没有了右滑返回手势。

鱼和熊掌不可兼得?笔者自有办法!

笔者尝试写个demo来验证有什么办法可以解决,尝试了以下四种:

  • 只在当前controller遵守UIGestureRecognizerDelegate并设置代理为self
  • 将UIGestureRecognizerDelegate放在公共基类控制器遵守并设置代理为self,然后子类重写代理方法
  • 将UIGestureRecognizerDelegate放在公共导航类HYBNavigationController里遵守,并设置代理为导航类,然后重写push/pop相关的所有方法
  • 将UIGestureRecognizerDelegate放在公共导航类HYBNavigationController里遵守,并设置代理为导航类,但是,只遵守-gestureRecognizerShouldBegin:代理方法

方案一(不可行)

方案一:只在当前controller遵守UIGestureRecognizerDelegate并设置代理为self

为什么不可行呢?当想不测试怎么知道呢?光想是很难考虑全面的。于是写了个小demo来测试。

我们在该controller里这样写:


   
   
  1. - (void)viewDidLoad { 
  2.  
  3.   [super viewDidLoad]; 
  4.  
  5.   
  6.  
  7.     UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom]; 
  8.  
  9.   [button setTitle:@"返回" forState:UIControlStateNormal]; 
  10.  
  11.   [button addTarget:self action:@selector(onBack) forControlEvents:UIControlEventTouchUpInside]; 
  12.  
  13.   [button sizeToFit]; 
  14.  
  15.   [button setTitleColor:[UIColor blueColor] forState:UIControlStateNormal]; 
  16.  
  17.   UIBarButtonItem *btnItem = [[UIBarButtonItem alloc] initWithCustomView:button]; 
  18.  
  19.   self.navigationItem.leftBarButtonItem = btnItem; 
  20.  
  21.   
  22.  
  23.   // 关键行 
  24.  
  25.   self.navigationController.interactivePopGestureRecognizer.delegate = self; 
  26.  
  27.  

一旦设置了代理为self,那么使用leftBarButtonItem后就可以实现点击回调,而且右滑手势还在。

但是,self.navigationController那可是导航控制器对象的的代理被修改当某个控制器对象了,当这个控制器类被释放后,那么代理就为nil了,如此就再也没有右滑返回手势了。

那么可能有人会想,在-viewDidAppear:里设置代理为self,在-viewDidDisappear:时设置代理成原来的代理对象呢?同样不可以。当A push到B,B push到C,然后从C返回后,代理就不再是最初的导航代理了。

所以,该方案不可行。

方案二(不可行)

方案二:将UIGestureRecognizerDelegate放在公共基类控制器遵守并设置代理为self,然后子类重写代理方法

笔者尝试将UIGestureRecognizerDelegate放在HYBBaseViewControlle里遵守,然后实现代理,默认返回YES,表示支持右滑返回。如果要让某个控制器不支持右滑返回或者在返回前先执行什么操作,可以通过重写此代理方法来实现。

当只在一个控制器里时,这是可以实现的。但是,当这个控制器被释放了以后,代理对象就变成了nil了,因此代理是对于导航条对象的,不属性单个控制器的。

方案三(可行,但复杂)

方案三:将UIGestureRecognizerDelegate放在公共导航类HYBNavigationController里遵守,并设置代理为导航类,然后重写push/pop相关的所有方法。

如实现如何下:


   
   
  1. //  HYBNavigationController.m 
  2.  
  3. //  NavRightPanGestureDemo 
  4.  
  5. // 
  6.  
  7. //  Created by huangyibiao on 16/2/22. 
  8.  
  9. //  Copyright © 2016年 huangyibiao. All rights reserved. 
  10.  
  11. // 
  12.  
  13.   
  14.  
  15. #import "HYBNavigationController.h" 
  16.  
  17. #import "HYBBaseViewController.h" 
  18.  
  19.   
  20.  
  21. @interface HYBNavigationController () 
  22.  
  23.   
  24.  
  25. @property (nonatomic, assign) BOOL enableRightGesture; 
  26.  
  27.   
  28.  
  29. @end 
  30.  
  31.   
  32.  
  33. @implementation HYBNavigationController 
  34.  
  35.   
  36.  
  37. - (void)viewDidLoad { 
  38.  
  39.   [super viewDidLoad]; 
  40.  
  41.   
  42.  
  43.   self.enableRightGesture = YES; 
  44.  
  45.   self.interactivePopGestureRecognizer.delegate = self; 
  46.  
  47.  
  48.   
  49.  
  50. - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer { 
  51.  
  52.   return self.enableRightGesture; 
  53.  
  54.  
  55.   
  56.  
  57. - (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated { 
  58.  
  59.   if ([viewController isKindOfClass:[HYBBaseViewController class]]) { 
  60.  
  61.     if ([viewController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) { 
  62.  
  63.       HYBBaseViewController *vc = (HYBBaseViewController *)viewController; 
  64.  
  65.       self.enableRightGesture = [vc gestureRecognizerShouldBegin]; 
  66.  
  67.     } 
  68.  
  69.   } 
  70.  
  71.   
  72.  
  73.   [super pushViewController:viewController animated:YES]; 
  74.  
  75.  
  76.   
  77.  
  78. - (NSArray *)popToRootViewControllerAnimated:(BOOL)animated { 
  79.  
  80.      self.enableRightGesture = YES; 
  81.  
  82.   return [super popToRootViewControllerAnimated:animated]; 
  83.  
  84.  
  85.   
  86.  
  87. - (UIViewController *)popViewControllerAnimated:(BOOL)animated { 
  88.  
  89.   if (self.viewControllers.count == 1) { 
  90.  
  91.     self.enableRightGesture = YES; 
  92.  
  93.   } else { 
  94.  
  95.     NSUInteger index = self.viewControllers.count - 2; 
  96.  
  97.     UIViewController *destinationController = [self.viewControllers objectAtIndex:index]; 
  98.  
  99.     if ([destinationController isKindOfClass:[HYBBaseViewController class]]) { 
  100.  
  101.       if ([destinationController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) { 
  102.  
  103.         HYBBaseViewController *vc = (HYBBaseViewController *)destinationController; 
  104.  
  105.         self.enableRightGesture = [vc gestureRecognizerShouldBegin]; 
  106.  
  107.       } 
  108.  
  109.     } 
  110.  
  111.   } 
  112.  
  113.   
  114.  
  115.   return [super popViewControllerAnimated:animated]; 
  116.  
  117.  
  118.   
  119.  
  120. - (NSArray *)popToViewController:(UIViewController *)viewController animated:(BOOL)animated { 
  121.  
  122.   if (self.viewControllers.count == 1) { 
  123.  
  124.     self.enableRightGesture = YES; 
  125.  
  126.   } else { 
  127.  
  128.     UIViewController *destinationController = viewController; 
  129.  
  130.     if ([destinationController isKindOfClass:[HYBBaseViewController class]]) { 
  131.  
  132.       if ([destinationController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) { 
  133.  
  134.         HYBBaseViewController *vc = (HYBBaseViewController *)destinationController; 
  135.  
  136.         self.enableRightGesture = [vc gestureRecognizerShouldBegin]; 
  137.  
  138.       } 
  139.  
  140.     } 
  141.  
  142.   } 
  143.  
  144.   
  145.  
  146.   return [super popToViewController:viewController animated:animated]; 
  147.  
  148.  
  149.   
  150.  
  151. @end  

这是通过重写所有的pop/push相关方法,通过判断是否要求支持右滑来设置。然后,我们要让某个控制器类在右滑返回或者点击返回之前,先调用我们的API判断,如下:


   
   
  1. #import "HYBBController.h" 
  2.  
  3.   
  4.  
  5. @implementation HYBBController 
  6.  
  7.   
  8.  
  9. - (void)viewDidLoad { 
  10.  
  11.   [super viewDidLoad]; 
  12.  
  13.   
  14.  
  15.   UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom]; 
  16.  
  17.   [button setTitle:@"返回" forState:UIControlStateNormal]; 
  18.  
  19.   [button addTarget:self action:@selector(onBack) forControlEvents:UIControlEventTouchUpInside]; 
  20.  
  21.   [button sizeToFit]; 
  22.  
  23.   [button setTitleColor:[UIColor blueColor] forState:UIControlStateNormal]; 
  24.  
  25.   UIBarButtonItem *btnItem = [[UIBarButtonItem alloc] initWithCustomView:button]; 
  26.  
  27.   self.navigationItem.leftBarButtonItem = btnItem; 
  28.  
  29.  
  30.   
  31.  
  32. - (BOOL)gestureRecognizerShouldBegin { 
  33.  
  34.   [self onBack]; 
  35.  
  36.   return NO
  37.  
  38.  
  39.   
  40.  
  41. - (void)onBack { 
  42.  
  43.   UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"标哥的技术博客" 
  44.  
  45.                                                       message:@"知道博客地址是什么吗?" 
  46.  
  47.                                                      delegate:self 
  48.  
  49.                                             cancelButtonTitle:@"不知道" 
  50.  
  51.                                             otherButtonTitles:@"知道", nil]; 
  52.  
  53.   [alertView show]; 
  54.  
  55.  
  56.   
  57.  
  58. #pragma mark - UIAlertViewDelegate 
  59.  
  60. - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex { 
  61.  
  62.   if (buttonIndex == 0) { 
  63.  
  64.   
  65.  
  66.   } else { 
  67.  
  68.     if ([self.navigationItem.title isEqualToString:@"VC6"]) { 
  69.  
  70.       NSUInteger index = self.navigationController.viewControllers.count - 3; 
  71.  
  72.       UIViewController *vc = [self.navigationController.viewControllers objectAtIndex:index]; 
  73.  
  74.       [self.navigationController popToViewController:vc animated:YES]; 
  75.  
  76.     } else { 
  77.  
  78.       [self.navigationController popViewControllerAnimated:YES]; 
  79.  
  80.     } 
  81.  
  82.   } 
  83.  
  84.  
  85.   
  86.  
  87. @end  

这种方案确实实现了我们的需求。但是,有没有更简单的方案呢?今天可能是眼睛有点困的原因,在研究的时候没有意识到第四种方案。在我准备写这篇文章的时候,我再认识地理了一遍逻辑,发现还有非常简单的一种方案可以实现我的需求。

方案四(可靠,最优)

方案四:将UIGestureRecognizerDelegate放在公共导航类HYBNavigationController里遵守,并设置代理为导航类,但是,只遵守-gestureRecognizerShouldBegin:代理方法。


   
   
  1. @interface HYBNavigationController () 
  2.  
  3.   
  4.  
  5. @end 
  6.  
  7.   
  8.  
  9. @implementation HYBNavigationController 
  10.  
  11.   
  12.  
  13. - (void)viewDidLoad { 
  14.  
  15.   [super viewDidLoad]; 
  16.  
  17.   
  18.  
  19.   self.interactivePopGestureRecognizer.delegate = self; 
  20.  
  21.  
  22.   
  23.  
  24. - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer { 
  25.  
  26.   BOOL ok = YES; // 默认为支持右滑反回 
  27.  
  28.   if ([self.topViewController isKindOfClass:[HYBBaseViewController class]]) { 
  29.  
  30.     if ([self.topViewController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) { 
  31.  
  32.       HYBBaseViewController *vc = (HYBBaseViewController *)self.topViewController; 
  33.  
  34.      ok = [vc gestureRecognizerShouldBegin]; 
  35.  
  36.     } 
  37.  
  38.   } 
  39.  
  40.   
  41.  
  42.   return ok; 
  43.  
  44.  
  45.   
  46.  
  47. @end  

使用方法与第三种方案一样,是不是非常地简化了?看来是元宵给我的礼物啊,突然想到这样的办法。以前一直没有研究过interactivePopGestureRecognizer属性,这个属性是iOS7以后才有的,因此在项目中一直不能直接使用leftBarButtonItem处理,除非那个界面不要右滑返回。

现在,一切都明了了,想要使用leftBarButtonItem在公共基类控制器中统一调用API来设置就非常简单了,右滑返回手势也可以正常使用~

还等什么,赶紧试试吧!

最后

如果你所使用的项目也有这样的需求,不防试试吧!笔者提供了demo的,因此可以先下载demo来看看效果哦!经过多次测试,笔者认为这是可行的方案,大家若在使用中出现问题,还请反馈与笔者,我也想了解是什么情况,当然也要找解决方案,共同进步嘛。





本文作者:佚名
来源:51CTO
目录
相关文章
|
1月前
|
安全 Android开发 数据安全/隐私保护
深入探讨iOS与Android系统安全性对比分析
在移动操作系统领域,iOS和Android无疑是两大巨头。本文从技术角度出发,对这两个系统的架构、安全机制以及用户隐私保护等方面进行了详细的比较分析。通过深入探讨,我们旨在揭示两个系统在安全性方面的差异,并为用户提供一些实用的安全建议。
|
29天前
|
安全 数据安全/隐私保护 Android开发
深入探索iOS系统安全机制:从基础到高级
本文旨在全面解析iOS操作系统的安全特性,从基础的权限管理到高级的加密技术,揭示苹果如何构建一个既开放又安全的移动平台。我们将通过实例和分析,探讨iOS系统如何保护用户数据免受恶意软件、网络攻击的威胁,并对比Android系统在安全性方面的差异。
|
1月前
|
安全 搜索推荐 Android开发
揭秘安卓与iOS系统的差异:技术深度对比
【10月更文挑战第27天】 本文深入探讨了安卓(Android)与iOS两大移动操作系统的技术特点和用户体验差异。通过对比两者的系统架构、应用生态、用户界面、安全性等方面,揭示了为何这两种系统能够在市场中各占一席之地,并为用户提供不同的选择。文章旨在为读者提供一个全面的视角,理解两种系统的优势与局限,从而更好地根据自己的需求做出选择。
80 2
|
2月前
|
安全 搜索推荐 Android开发
深入探索安卓与iOS系统的差异及其对用户体验的影响
在当今的智能手机市场中,安卓和iOS是两大主流操作系统。它们各自拥有独特的特性和优势,为用户提供了不同的使用体验。本文将深入探讨安卓与iOS系统之间的主要差异,包括它们的设计理念、用户界面、应用生态以及安全性等方面,并分析这些差异如何影响用户的使用体验。
|
2月前
|
安全 搜索推荐 Android开发
揭秘iOS与Android系统的差异:一场技术与哲学的较量
在当今数字化时代,智能手机操作系统的选择成为了用户个性化表达和技术偏好的重要标志。iOS和Android,作为市场上两大主流操作系统,它们之间的竞争不仅仅是技术的比拼,更是设计理念、用户体验和生态系统构建的全面较量。本文将深入探讨iOS与Android在系统架构、应用生态、用户界面及安全性等方面的本质区别,揭示这两种系统背后的哲学思想和市场策略,帮助读者更全面地理解两者的优劣,从而做出更适合自己的选择。
|
1月前
|
安全 Android开发 iOS开发
深入探索iOS与Android系统的差异性及优化策略
在当今数字化时代,移动操作系统的竞争尤为激烈,其中iOS和Android作为市场上的两大巨头,各自拥有庞大的用户基础和独特的技术特点。本文旨在通过对比分析iOS与Android的核心差异,探讨各自的优势与局限,并提出针对性的优化策略,以期为用户提供更优质的使用体验和为开发者提供有价值的参考。
|
1月前
|
安全 Android开发 iOS开发
安卓系统与iOS系统的比较####
【10月更文挑战第26天】 本文将深入探讨安卓(Android)和iOS这两大主流移动操作系统的各自特点、优势与不足。通过对比分析,帮助读者更好地理解两者在用户体验、应用生态、系统安全等方面的差异,从而为消费者在选择智能手机时提供参考依据。无论你是技术爱好者还是普通用户,这篇文章都将为你揭示两大系统背后的故事和技术细节。 ####
47 0
|
2月前
|
IDE Android开发 iOS开发
探索安卓与iOS系统的技术差异:开发者的视角
本文深入分析了安卓(Android)与苹果iOS两大移动操作系统在技术架构、开发环境、用户体验和市场策略方面的主要差异。通过对比这两种系统的不同特点,旨在为移动应用开发者提供有价值的见解,帮助他们在不同平台上做出更明智的开发决策。
|
3月前
|
开发工具 Swift 数据安全/隐私保护
移动应用开发之旅:从零到一的iOS系统探索
【9月更文挑战第16天】在数字时代的浪潮中,移动应用成为连接用户与数字世界的桥梁。本文将带你走进iOS移动操作系统的世界,了解其架构、设计理念以及开发环境。我们将通过Swift语言的简单示例,展示如何构建一个基本的iOS应用,并探讨移动应用开发的未来趋势。无论你是编程新手还是资深开发者,这篇文章都将为你提供宝贵的见解和知识。
|
3月前
|
监控 Android开发 iOS开发
深入探索安卓与iOS的系统架构差异:理解两大移动平台的技术根基在移动技术日新月异的今天,安卓和iOS作为市场上最为流行的两个操作系统,各自拥有独特的技术特性和庞大的用户基础。本文将深入探讨这两个平台的系统架构差异,揭示它们如何支撑起各自的生态系统,并影响着全球数亿用户的使用体验。
本文通过对比分析安卓和iOS的系统架构,揭示了这两个平台在设计理念、安全性、用户体验和技术生态上的根本区别。不同于常规的技术综述,本文以深入浅出的方式,带领读者理解这些差异是如何影响应用开发、用户选择和市场趋势的。通过梳理历史脉络和未来展望,本文旨在为开发者、用户以及行业分析师提供有价值的见解,帮助大家更好地把握移动技术发展的脉络。
104 6