需求场景
需求:APP中经常有一些页面,有很多的填写,如果用户不小心触发了返回,这些资料都没有保存会非常影响体验;或者一些功能有多个页面流程,不希望返回上一页,而是要做某些处理或返回首页。这种情况我们就要对返回手势和按钮做些处理,如图
比较直接的方式就是设置返回按钮。但是有个弊端,原生手势返回也失效了,如果要手势返回,处理不好还有可能产生假卡死的BUG(显示的页面卡着不动,但实际已经是其他页面,按Home键,再打开APP就会显示真实页面)。直接设置如下代码
UIButton *backButton = [[UIButton alloc]initWithFrame:CGRectMake(0, 0, 40, 40)];
[backButton setImage:[UIImage imageNamed:@"back"] forState:UIControlStateNormal];
[backButton addTarget:self action:@selector(backItemClick) forControlEvents:UIControlEventTouchUpInside];
UIBarButtonItem *backItem = [[UIBarButtonItem alloc]initWithCustomView:backButton];
self.navigationItem.leftBarButtonItem = backItem;
这样总不能每个需要的页面都这样写,那样太麻烦。我们肯定是希望想拦截的时候实现某个方法就行。
关键思路的代码
1.原生的返回按钮,点击的时候其实会调用一个私有的方法,可以考虑重写这个方法(可以通过runtime机制将方法的列表打印出来看看,这里不演示)
- (BOOL)navigationBar:(UINavigationBar *)navigationBar shouldPopItem:(UINavigationItem *)item
2.既然重写返回按钮,手势会被禁用,那么干脆禁用掉系统手势,自定义一个手势。大概的逻辑就是通过runtime机制,交换掉pushViewControlle方法,为push的页面添加一个手势,通过KVC取出原生那个手势的target和sel。
- (void)hzs_pushViewController:(UIViewController *)viewController animated:(BOOL)animated
{
if (![self.interactivePopGestureRecognizer.view.gestureRecognizers containsObject:self.hzs_backGestureRecognizer])
{
/*
这里逻辑是禁用原来的手势返回,并运用kvc取出target 和action交给自定义的手势响应
*/
[self.interactivePopGestureRecognizer.view addGestureRecognizer:self.hzs_backGestureRecognizer];
NSArray *internalTargets = [self.interactivePopGestureRecognizer valueForKey:@"targets"];
id internalTarget = [internalTargets.firstObject valueForKey:@"target"];
SEL internalAction = NSSelectorFromString(@"handleNavigationTransition:");
self.hzs_backGestureRecognizer.delegate = self;
[self.hzs_backGestureRecognizer addTarget:internalTarget action:internalAction];
self.interactivePopGestureRecognizer.enabled = NO;
}
if (![self.viewControllers containsObject:viewController]) {
[self hzs_pushViewController:viewController animated:animated];
}
}
由于整体代码贴上来就太影响阅读,所以这里就不贴全部代码。为了方便我以后的使用,我支持了cocoapods安装
demo地址:https://github.com/1498522607/HZSHookVCBack
使用方法
pod 'HZSHookVCBack' 安装
导入头文件HZSHookVCBack.h ,然后在需要拦截的页面实现UINavigationControllerHookBackDelegate的方法就行,已经遵守协议,只要在ViewController添加一下代码即可
- (BOOL)hzs_backGestureAction {
NSLog(@"点击了返回按钮");
return NO;
}
- (BOOL)hzs_backBarButtonItemDidClickAction {
NSLog(@"触发了返回手势");
return NO;
}
最后说明
文章和代码有什么问题请给我留言,后期我会更正的