UINavigationController返回手势失效问题

从iOS7开始,系统为UINavigationController提供了一个interactivePopGestureRecognizer用于右滑返回(pop),但是,如果自定了back button或者隐藏了navigationBar,该手势就失效了。

image

这是为什么呢?

原因

我们知道,interactivePopGestureRecognizer从手势触发到行为发生,要经过下面的阶段:

image

interactivePopGestureRecognizer还存在,但没有起作用,可能是delegate里被阻断了没调用target/action,或者是调用了target/action没运行动画。

经过尝试(参考别人博客),发现自定义返回按钮或者隐藏navigationBar导致的该手势未起作用是因为在delegate阶段被阻断了。

如果我们知道action的名字,则可以添加一个自定义的滑动手势,直接调用该系统action。但API文档并没有提供。

结果就是要么自己实现滑动返回的动画action,要么自己重写interactivePopGestureRecognizer的delegate以让手势继续下去,触发系统的动画action。

实现方法

那就把delegate自己实现一下吧。

新建一个类BaseNavigationController,实现delegate:

1
2
3
4
- (void)viewDidLoad {
    [super viewDidLoad];
    self.interactivePopGestureRecognizer.delegate =  self;
}

让手势生效

1
2
3
4
5
6
7
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer {
    if (self.viewControllers.count <= 1 ) {
        return NO;
    }

    return YES;
}

在需要滑动返回的地方的UINavigationController换成BaseNavigationController

完成!

image

然后考虑到在push动画发生的时候,禁止滑动手势,在BaseNavigationController添加

1
2
3
4
- (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated {
    [super pushViewController:viewController animated:animated];
    self.interactivePopGestureRecognizer.enabled = NO;
}

在使用navigationController的viewcontroller里添加

1
2
3
4
- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:animated];
    self.navigationController.interactivePopGestureRecognizer.enabled = YES;
}

结果

当然不需要把上面的代码都抄一遍,因为这么通用的功能由一位韩国开发者做成组件,放在了github https://github.com/devxoul/SwipeBack。你只需要

1
2
platform :ios, '7.0'
pod 'SwipeBack', '~> 1.0'

该工程用了category+JRSwizzle交换了上面涉及到的UINavigationController的那些方法,还额外考虑了只自定义back button而不隐藏navigationBar的情况。无需一行代码,让系统的右滑返回动画重新回来!


原文地址:http://chisj.github.io/blog/2015/05/27/uinavigationcontrollerfan-hui-shou-shi-shi-xiao-wen-ti/



  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 iOS 开发中,可以通过实现 UINavigationControllerDelegate 协议中的 navigationController:didShowViewController:animated: 方法来拦截侧滑返回手势。 例如,你可以在这个方法中判断当前展示的视图控制器是否需要禁用侧滑返回手势,如果需要,则设置导航控制器的 interactivePopGestureRecognizer.enabled 属性为 NO,否则设置为 YES。 示例代码如下: ``` - (void)navigationController:(UINavigationController *)navigationController didShowViewController:(UIViewController *)viewController animated:(BOOL)animated { BOOL shouldEnableGesture = YES; if ([viewController conformsToProtocol:@protocol(DisablePopGestureProtocol)]) { shouldEnableGesture = [viewController disablePopGesture]; } navigationController.interactivePopGestureRecognizer.enabled = shouldEnableGesture; } ``` 在这段代码中,我们新增了一个协议 DisablePopGestureProtocol,用于标记那些视图控制器需要禁用侧滑返回手势。在每个视图控制器中,如果希望禁用侧滑返回手势,则只需要实现这个协议中的 disablePopGesture 方法并返回 NO 即可。 希望这些信息对你有所帮助! ### 回答2: 在iOS开发中,拦截侧滑返回手势的方法主要是通过自定义导航控制器或者使用UINavigationControllerDelegate的相关方法来实现。 一种方法是创建一个自定义导航控制器,并在其中重写`interactivePopGestureRecognizer`的`delegate`属性。首先,我们需要创建一个继承自`UINavigationController`的子类,比如`CustomNavigationController`,然后重写其`viewDidLoad`方法,在该方法内设置`interactivePopGestureRecognizer`的`delegate`属性为自身,并实现`UIGestureRecognizerDelegate`协议,最后在`gestureRecognizerShouldBegin`方法内返回`false`。这样一来,在默认的滑动返回手势中,只要是继承自这个自定义导航控制器的视图控制器都将拦截滑动返回手势,无法触发返回动作。 另一种方法是通过实现`UINavigationControllerDelegate`协议相关方法来拦截侧滑返回手势。在视图控制器中设置`UINavigationController`的代理为自身,并实现`navigationController(_:willShow:animated:)`方法。在该方法内,我们可以通过修改导航控制器的`interactivePopGestureRecognizer`属性的`enabled`属性来控制是否允许滑动返回手势。当我们设置为`false`时,滑动返回手势将被拦截,无法触发返回动作。 需要注意的是,如果选择第二种方法,我们还需要在视图控制器将要消失的时候将导航控制器的代理置空,以避免内存泄漏和不必要的代理方法调用。 总结起来,要拦截侧滑返回手势,可以通过自定义导航控制器或者使用`UINavigationControllerDelegate`的相关方法,来对滑动返回手势进行监听和控制。 ### 回答3: 在iOS开发中,拦截侧滑返回手势可以通过以下几个步骤实现。 首先,我们可以通过遵循UIGestureRecognizerDelegate协议,并实现它的gestureRecognizerShouldBegin方法。这个方法会在每次手势开始时被调用。 在实现gestureRecognizerShouldBegin方法时,可以通过判断当前viewController的类型或者一些其他的条件,来决定是否拦截侧滑返回手势。如果不满足拦截条件,可以返回NO,表示不拦截手势,否则返回YES。 例如,想要在特定的viewController中拦截侧滑返回手势,可以按照以下方式实现: ``` extension ViewController: UIGestureRecognizerDelegate { override func viewDidLoad() { super.viewDidLoad() navigationController?.interactivePopGestureRecognizer?.delegate = self } func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool { if let currentVC = navigationController?.topViewController, currentVC is MySpecialViewController { return false } return true } } ``` 在上述代码中,将ViewController设置为UIGestureRecognizerDelegate的代理,然后在viewDidLoad方法中,将导航控制器的interactivePopGestureRecognizer的代理设置为self,即ViewController。接着,在gestureRecognizerShouldBegin方法中,判断当前导航控制器的顶层视图控制器是否是特定的MySpecialViewController,并返回相应的布尔值。 这样,如果当前在MySpecialViewController中,就会拦截侧滑返回手势。 以上就是在iOS开发中拦截侧滑返回手势的方法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值