1.问题原因:
iOS在根视图触发返回手势的时候,可能会导致界面卡死,为了解决这个问题,主要的思路就是在根视图上关闭返回手势。
2.解决办法:
- 创建自定义导航控制器,设置代理为自己:
- (void)viewDidLoad {
[super viewDidLoad];
@weakiy(self);
self.delegate = weak_self;
}
- 实现代理方法,在控制器didShow的时候,检查viewControllers里的元素是否只有一个了。如果只有一个,这个时候就要关掉返回手势,否则此时再返回就要触发界面卡死的问题了 (重点!!!),
- (void)navigationController:(UINavigationController *)navigationController didShowViewController:(UIViewController *)viewController animated:(BOOL)animated
{
if ([navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)])
{
NSString * className = NSStringFromClass([navigationController.viewControllers.lastObject class]);
if (navigationController.viewControllers.count == 1)
{
//根视图控制器关闭手势返回
navigationController.interactivePopGestureRecognizer.enabled = NO;
navigationController.interactivePopGestureRecognizer.delegate = nil;
}else
{
//其他的页面默认开启手势返回
navigationController.interactivePopGestureRecognizer.enabled = YES;
}
NSLog(@"**************************************1,%@",className);
}
}
3.补充
- 这里有个小技巧值得说一下。通常有些页面可能会需要关闭手势返回,这个时候可以在这里引入一个黑名单的机制,说白了就是一个数组,将你想要屏蔽的页面事先存在这个数组里,类似下面这样:
- (NSArray *)blacklist
{
//黑名单中的类禁止手势返回
if (!_blacklist)
{
_blacklist = @[@"FirstViewController",
@"SecondViewController",
@"ThirdViewController"];
}
return _blacklist;
}
- 然后结合上面的代理方法,就是这个样子:
- (void)navigationController:(UINavigationController *)navigationController didShowViewController:(UIViewController *)viewController animated:(BOOL)animated
{
if ([navigationController respondsToSelector:@selector(interactivePopGestureRecognizer)])
{
NSString * className = NSStringFromClass([navigationController.viewControllers.lastObject class]);
if (navigationController.viewControllers.count == 1)
{
//根视图控制器关闭手势返回
navigationController.interactivePopGestureRecognizer.enabled = NO;
navigationController.interactivePopGestureRecognizer.delegate = nil;
}else if ([self.blacklist containsObject:className])
{
//黑名单中的控制器关闭手势返回
navigationController.interactivePopGestureRecognizer.enabled = NO;
navigationController.interactivePopGestureRecognizer.delegate = nil;
}else
{
//其他的页面默认开启手势返回
navigationController.interactivePopGestureRecognizer.enabled = YES;
}
NSLog(@"**************************************1,%@",className);
}
}
- 利用NSStringFromClass函数,检查当前出现的页面是否在黑名单中,将黑名单中的控制器关闭手势,这样就可以在不引入头文件的情况下实现宏观控制单个页面的侧滑手势的开关
4.注意事项
这个方法其实还是有缺点的,比如将导航控制器的代理设置成自己了,那么这个时候就不能再变更代理,否则上面这些操作就不会生效。(虽然这些代理方法平时我也不怎么用到)。但是,这里还是给出两个解决方案吧
- 方案一:将你需要的协议方法写成block,在这自定义的导航控制器中实现,再通过block输出
- 方案二:还是在这自定义的导航控制器中实现你需要的协议方法,然后将你在收到协议方法回调后需要实现的方法,暴露到那个控制器的头文件中,然后在viewControllers中找到那个对象,直接调他的方法;