iOS系统右滑返回全局控制方案
前言
今天有个小需求,在点击导航条上的返回按钮之前要调用某个API,并弹出UIAlertView来显示,根据用户的选项判断是否是返回还是继续留在当前控制器。举个简单的例子,当点击导航条上的左上角返回按钮时,就调用我们的API来提示是否知道,点击知道则返回,点击不知道则继续留在当前控制器。
那么问题来了,导航自带的右滑返回手势在点击系统的返回按钮时,不会没有办法处理,那是自动的,因此就要想办法改成leftBarButtonItem了,但是使用了leftBarButtonItem就没有了右滑返回手势。
鱼和熊掌不可兼得?笔者自有办法!
笔者尝试写个demo来验证有什么办法可以解决,尝试了以下四种:
- 只在当前controller遵守UIGestureRecognizerDelegate并设置代理为self
- 将UIGestureRecognizerDelegate放在公共基类控制器遵守并设置代理为self,然后子类重写代理方法
- 将UIGestureRecognizerDelegate放在公共导航类HYBNavigationController里遵守,并设置代理为导航类,然后重写push/pop相关的所有方法
- 将UIGestureRecognizerDelegate放在公共导航类HYBNavigationController里遵守,并设置代理为导航类,但是,只遵守-gestureRecognizerShouldBegin:代理方法
方案一(不可行)
方案一:只在当前controller遵守UIGestureRecognizerDelegate并设置代理为self
为什么不可行呢?当想不测试怎么知道呢?光想是很难考虑全面的。于是写了个小demo来测试。
我们在该controller里这样写:
- (void)viewDidLoad {
[super viewDidLoad];
UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
[button setTitle:@"返回" forState:UIControlStateNormal];
[button addTarget:self action:@selector(onBack) forControlEvents:UIControlEventTouchUpInside];
[button sizeToFit];
[button setTitleColor:[UIColor blueColor] forState:UIControlStateNormal];
UIBarButtonItem *btnItem = [[UIBarButtonItem alloc] initWithCustomView:button];
self.navigationItem.leftBarButtonItem = btnItem;
// 关键行
self.navigationController.interactivePopGestureRecognizer.delegate = self;
}
一旦设置了代理为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相关的所有方法。
如实现如何下:
//
// HYBNavigationController.m
// NavRightPanGestureDemo
//
// Created by huangyibiao on 16/2/22.
// Copyright © 2016年 huangyibiao. All rights reserved.
// #import "HYBNavigationController.h"
#import "HYBBaseViewController.h" @interface HYBNavigationController () <UIGestureRecognizerDelegate> @property (nonatomic, assign) BOOL enableRightGesture; @end @implementation HYBNavigationController - (void)viewDidLoad {
[super viewDidLoad]; self.enableRightGesture = YES;
self.interactivePopGestureRecognizer.delegate = self;
} - (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer {
return self.enableRightGesture;
} - (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated {
if ([viewController isKindOfClass:[HYBBaseViewController class]]) {
if ([viewController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
HYBBaseViewController *vc = (HYBBaseViewController *)viewController;
self.enableRightGesture = [vc gestureRecognizerShouldBegin];
}
} [super pushViewController:viewController animated:YES];
} - (NSArray<UIViewController *> *)popToRootViewControllerAnimated:(BOOL)animated {
self.enableRightGesture = YES;
return [super popToRootViewControllerAnimated:animated];
} - (UIViewController *)popViewControllerAnimated:(BOOL)animated {
if (self.viewControllers.count == ) {
self.enableRightGesture = YES;
} else {
NSUInteger index = self.viewControllers.count - ;
UIViewController *destinationController = [self.viewControllers objectAtIndex:index];
if ([destinationController isKindOfClass:[HYBBaseViewController class]]) {
if ([destinationController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
HYBBaseViewController *vc = (HYBBaseViewController *)destinationController;
self.enableRightGesture = [vc gestureRecognizerShouldBegin];
}
}
} return [super popViewControllerAnimated:animated];
} - (NSArray<UIViewController *> *)popToViewController:(UIViewController *)viewController animated:(BOOL)animated {
if (self.viewControllers.count == ) {
self.enableRightGesture = YES;
} else {
UIViewController *destinationController = viewController;
if ([destinationController isKindOfClass:[HYBBaseViewController class]]) {
if ([destinationController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
HYBBaseViewController *vc = (HYBBaseViewController *)destinationController;
self.enableRightGesture = [vc gestureRecognizerShouldBegin];
}
}
} return [super popToViewController:viewController animated:animated];
} @end
这是通过重写所有的pop/push相关方法,通过判断是否要求支持右滑来设置。然后,我们要让某个控制器类在右滑返回或者点击返回之前,先调用我们的API判断,如下:
#import "HYBBController.h"
@implementation HYBBController
- (void)viewDidLoad {
[super viewDidLoad];
UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
[button setTitle:@"返回" forState:UIControlStateNormal];
[button addTarget:self action:@selector(onBack) forControlEvents:UIControlEventTouchUpInside];
[button sizeToFit];
[button setTitleColor:[UIColor blueColor] forState:UIControlStateNormal];
UIBarButtonItem *btnItem = [[UIBarButtonItem alloc] initWithCustomView:button];
self.navigationItem.leftBarButtonItem = btnItem;
}
- (BOOL)gestureRecognizerShouldBegin {
[self onBack];
return NO;
}
- (void)onBack {
UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"标哥的技术博客"
message:@"知道博客地址是什么吗?"
delegate:self
cancelButtonTitle:@"不知道"
otherButtonTitles:@"知道", nil];
[alertView show];
}
#pragma mark - UIAlertViewDelegate
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {
if (buttonIndex == ) {
} else {
if ([self.navigationItem.title isEqualToString:@"VC6"]) {
NSUInteger index = self.navigationController.viewControllers.count - ;
UIViewController *vc = [self.navigationController.viewControllers objectAtIndex:index];
[self.navigationController popToViewController:vc animated:YES];
} else {
[self.navigationController popViewControllerAnimated:YES];
}
}
}
@end
这种方案确实实现了我们的需求。但是,有没有更简单的方案呢?今天可能是眼睛有点困的原因,在研究的时候没有意识到第四种方案。在我准备写这篇文章的时候,我再认识地理了一遍逻辑,发现还有非常简单的一种方案可以实现我的需求。
方案四(可靠,最优)
方案四:将UIGestureRecognizerDelegate放在公共导航类HYBNavigationController里遵守,并设置代理为导航类,但是,只遵守-gestureRecognizerShouldBegin:代理方法。
@interface HYBNavigationController () <UIGestureRecognizerDelegate>
@end
@implementation HYBNavigationController
- (void)viewDidLoad {
[super viewDidLoad];
self.interactivePopGestureRecognizer.delegate = self;
}
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer {
BOOL ok = YES; // 默认为支持右滑反回
if ([self.topViewController isKindOfClass:[HYBBaseViewController class]]) {
if ([self.topViewController respondsToSelector:@selector(gestureRecognizerShouldBegin)]) {
HYBBaseViewController *vc = (HYBBaseViewController *)self.topViewController;
ok = [vc gestureRecognizerShouldBegin];
}
}
return ok;
}
@end
使用方法与第三种方案一样,是不是非常地简化了?看来是元宵给我的礼物啊,突然想到这样的办法。以前一直没有研究过interactivePopGestureRecognizer属性,这个属性是iOS7以后才有的,因此在项目中一直不能直接使用leftBarButtonItem处理,除非那个界面不要右滑返回。
现在,一切都明了了,想要使用leftBarButtonItem在公共基类控制器中统一调用API来设置就非常简单了,右滑返回手势也可以正常使用~
还等什么,赶紧试试吧!
最后
如果你所使用的项目也有这样的需求,不防试试吧!笔者提供了demo的,因此可以先下载demo来看看效果哦!经过多次测试,笔者认为这是可行的方案,大家若在使用中出现问题,还请反馈与笔者,我也想了解是什么情况,当然也要找解决方案,共同进步嘛。
源代码
请大家到GITHUB下载吧:CoderJackyHuang
iOS系统右滑返回全局控制方案的更多相关文章
- iOS页面右滑返回的实现方法总结
1.边缘触发的系统方法 ①系统返回按钮 self.navigationController.interactivePopGestureRecognizer.enabled = YES; ②自定义返回 ...
- 第二十六篇、因为自定item(nav)而使系统右滑返回手势失效的解决方法
@interface ViewController () <uigesturerecognizerdelegate> @end@implementation ViewController ...
- iOS 开发 右滑返回上一级控制器
#import <objc/runtime.h> @interface UINavigationController (Transition)<UIGestureRecognizer ...
- iOS开发之自定义导航栏返回按钮右滑返回手势失效的解决
我相信针对每一个iOS开发者来说~除了根视图控制器外~所有的界面通过导航栏push过去的界面都是可以通过右滑来返回上一个界面~其实~在很多应用和APP中~用户已经习惯了这个功能~然而~作为开发者的我们 ...
- 全新的手势,侧滑返回、全局右滑返回都OUT啦!
前言 Android快速开发框架-ZBLibrary 最近将以前的 全局右滑返回 手势功能改成了 底部左右滑动手势. 为什么呢?为了解决滑动返回手势的问题. 目前有3种滑动返回手势 一.侧滑返回 代表 ...
- App开发流程之右滑返回手势功能续
上一篇记录了利用系统私有变量和方法实现右滑返回手势功能:http://www.cnblogs.com/ALongWay/p/5893515.html 这篇继续记录另一种方案:利用UINavigatio ...
- App开发流程之右滑返回手势功能
iOS7以后,导航控制器,自带了从屏幕左边缘右滑返回的手势功能. 但是,如果自定义了导航栏返回按钮,这项功能就失效了,需要自行实现.又如果需要修改手势触发范围,还是需要自行实现. 广泛应用的一种实现方 ...
- ios7自定义返回按钮后,右滑返回功能失效解决方法
-(void)viewWillAppear:(BOOL)animated{ [super viewWillAppear:animated]; //开启ios右滑返回 if ([ ...
- 想要隐藏navigationBar,同时又想支持右滑返回功能
如果直接设置 self.navigationBarHidden = YES; 那同时也会屏蔽右滑返回功能. 解决办法1: self.navigationBarHidden = NO; self.nav ...
随机推荐
- 多级联动导航栏(top导航)
http://www.dynamicdrive.com/style/csslibrary/item/jquery_multi_level_css_menu_2/ This is a multi-lev ...
- SCAU 1138 代码等式
1138 代码等式 时间限制:500MS 内存限制:65536K提交次数:59 通过次数:21 题型: 编程题 语言: 无限制 Description 一个代码等式就是形如x1x2...xi=y ...
- wine的中文字体显示
从1.1.4开始wine的界面就已经支持中文了,但是对于软件中的中文支持并不太好,主要原因.还是字体...Let's go 首先,copy一下字体:把simsun.ttc (即宋体)复制到 ~/.wi ...
- jquery validate easyui tooltip
jquery validate.js onfocusin: function (element, event) { this.lastActive = element; // hide error l ...
- Spark RDD概念学习系列之细谈RDD的弹性(十六)
细谈RDD的弹性 所谓,弹性,是指在内存不够时可以与磁盘进行交换. 弹性之一:自动的进行内存和磁盘数据存储的切换 弹性之二:基于Lineage(血缘)的高效容错 弹性之三:Task如果失败会 ...
- HDU 5660 jrMz and angles (暴力枚举)
jrMz and angles 题目链接: http://acm.hust.edu.cn/vjudge/contest/123316#problem/E Description jrMz has tw ...
- 扩展KMP题目
hdu4333 /* 题意:字符串s[0..n-1],每次把最后一个字符放到前面,求形成的字符串比最初串分别小,相同,大于的个数 因为是为了练习扩展KMP所以肯定是扩展KMP, 为了循环方便,在后面复 ...
- PID参数整定快速入门(调节器参数整定方法)
PID调节器参数整定方法很多,常见的工程整定方法有临界比例度法.衰减曲线法和经验法.云润仪表以图文形式分别介绍调节器参数整定方法. 临界比例度法一个调节系统,在阶跃干扰作用下,出现既不发散也不衰减的等 ...
- Linux系统管理员应该了解的一些I/O统计工具
作为一个Linux系统管理员,统计各类IO是一项必不可少的工作.其统计工具中iostat显然又是最重要的一个统计手段.但是这里iostat不是本文的重点,因为这个工具的使用在网络上已经有大量的教程,可 ...
- 【Cocos2d-X开发学习笔记】第21期:动画类(CCAnimate)的使用
本系列学习教程使用的是cocos2d-x-2.1.4(最新版为3.0alpha0-pre) ,PC开发环境Windows7,C++开发环境VS2010 之前我们已经学习过一些方法让节点“动”起来,Co ...