iOS 侧滑返回详解FDFullscreenPopGesture
1、侧滑本身是系统自带的,但是是左边缘触发的侧滑;
2、是什么导致了自带的实现不起作用的?原因有
(1)、隐藏了系统右上角返回的按钮,系统会自动关掉左滑返回的功能;
(2)、禁用侧滑功能;
(3)、自定义返回键或者导航栏后,侧滑失灵;
3、怎么解决?
使用FDFullscreenPopGesture可以解决
一个支持全屏侧滑的 UINavigationController的类目。FDFullscreenPopGesture的这个类目:UINavigation+FDFullscreenPopGesture,扩展了iOS7+的侧滑返回功能。取代从左边缘滑起的是,我们可以从屏幕的任意位置滑动,并且伴随着返回上个界面的过渡动画也可以正常流畅。
这个框架中只有两个文件,添加到项目工程中,所有的导航控制器都将自动会支持全屏侧滑。
禁用导航控制器的侧滑功能:
navigationController.fd_fullscreenPopGestureRecognizer.enabled = NO;
禁用viewcontroller的侧滑功能:
viewController.fd_interactivePopDisabled = YES;
基于导航栏外形的viewcontroller:
当使用全屏侧滑功能时,它可以正确的处理导航栏的过渡动画。
- 有导航栏到无导航栏;
- 无导航栏到有导航栏;
- 无导航栏到无导航栏;
这种优化是默认开启的,我们不必再调用 UINavigationController的-setNavigationBarHidden:animated:方法了,而是使用viewcontroller的API来隐藏导航栏。
- (void)viewDidLoad {
[super viewDidLoad];
self.fd_prefersNavigationBarHidden = NO;
}
也就是说fd_prefersNavigationBarHidden这个属性,默认是No。
如果是带scrollView的视图控制器
如果你想在有scrollView或者scrollView的子类中,使用全屏侧滑,那么可以通过在自定义的scrollView或者scrollView的子类中,复写下面这个方法:
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer {
if (self.contentOffset.x <= 0) {
if ([otherGestureRecognizer.delegate isKindOfClass:NSClassFromString(@"_FDFullscreenPopGestureRecognizerDelegate")]) {
return YES;
}
}
return NO;
}
这个第三方框架FDFullscreenPopGesture中,定义了的UINavigationController+FDFullscreenPopGesture文件,其中分别包含对 UINavigationController和UIViewController的分类。
UINavigationController+FDFullscreenPopGesture.h文件:
1、在UINavigationController的分类 UINavigationController + FDFullscreenPopGesture中,
暴露给用户的属性有:
- @property (nonatomic, strong, readonly) UIPanGestureRecognizer *fd_fullscreenPopGestureRecognizer;//这个滑动手势,处理用户的侧滑弹出操作。
- @property (nonatomic, assign) BOOL fd_viewControllerBasedNavigationBarAppearanceEnabled;//判断fd_viewControllerBasedNavigationBarAppearanceEnabled属性,视图控制器可以自己控制,而不是全局的方式控制,导航栏的外观。,默认是YES。
2、在UIViewController的分类 UIViewController + FDFullscreenPopGesture中,允许视图控制器禁用用户弹出手势,比如某些情况下,视图控制器需要自己处理滑动手势。
其中,暴露给用户的属性有
- @property (nonatomic, assign) BOOL fd_interactivePopDisabled;//当处于导航控制栈中时,是否禁用用户弹出手势;
- @property (nonatomic, assign) BOOL fd_prefersNavigationBarHidden;//通过检查基于导航栏的视图控制器是否可用,来指明视图控制器的导航栏隐藏与否。默认是NO,一般导航栏是展示的。
- @property (nonatomic, assign) CGFloat fd_interactivePopMaxAllowedInitialDistanceToLeftEdge;//最大允许触发侧滑弹出手势的距离左边缘的距离。默认是0。忽略这一限制。
UINavigationController+FDFullscreenPopGesture.m文件:
1、 _FDFullscreenPopGestureRecognizerDelegate声明
这个内部类遵从 UIGestureRecognizerDelegate协议:
- @property (nonatomic, weak) UINavigationController *navigationController;
2、_FDFullscreenPopGestureRecognizerDelegate类实现
这个类实现了UIGestureRecognizerDelegate的代理方法:
_FDFullscreenPopGestureRecognizerDelegate实现.jpg
3、UIViewController的分类( UIViewController (FDFullscreenPopGesturePrivate) )的声明和实现的方法:
在分类中声明了 _FDViewControllerWillAppearInjectBlock的Block变量。
- + (void)load;
- 在这个方法中,将 viewWillAppear:和fd_viewWillAppear: 以及 viewWillDisappear:和fd_viewWillDisappear:做了运行时交换。
- 具体fd_viewWillAppear:和fd_viewWillDisappear:是做什么的?继续往下看,
- - (void)fd_viewWillAppear:(BOOL)animated
- - (void)fd_viewWillDisappear:(BOOL)animated
//fd_willAppearInjectBlock的getter/setter方法
- - (_FDViewControllerWillAppearInjectBlock)fd_willAppearInjectBlock
- - (void)setFd_willAppearInjectBlock:(_FDViewControllerWillAppearInjectBlock)block
4、 UINavigationController + FDFullscreenPopGesture分类
- + (void)load
- - (void)fd_pushViewController:(UIViewController *)viewController animated:(BOOL)animated
1、添加自己的手势到边缘侧滑手势依附的View上;
2、将原来的边缘侧滑手势的target和action添加到我们自己的手势上;
3、禁用原来的边缘侧滑手势; - - (void)fd_setupViewControllerBasedNavigationBarAppearanceIfNeeded:(UIViewController *)appearingViewController
1、判断是不是自己控制导航栏的是否隐藏;
2、如果是全局控制的,就直接返回;如果是自己控制的,就继续;
3、将控制导航栏的是否隐藏的Block赋值给即将出现的和即将消失的视图控制器。 - - (_FDFullscreenPopGestureRecognizerDelegate *)fd_popGestureRecognizerDelegate
//fd_popGestureRecognizerDelegate的getter方法 - - (UIPanGestureRecognizer *)fd_fullscreenPopGestureRecognizer
//fd_fullscreenPopGestureRecognizer的getter方法 - - (BOOL)fd_viewControllerBasedNavigationBarAppearanceEnabled
//fd_viewControllerBasedNavigationBarAppearanceEnabled的getter方法 - - (void)setFd_viewControllerBasedNavigationBarAppearanceEnabled:(BOOL)enabled
//fd_viewControllerBasedNavigationBarAppearanceEnabled的setter方法
5、 UIViewController + FDFullscreenPopGesture分类
//fd_interactivePopDisabled属性的getter/setter方法 - - (BOOL)fd_interactivePopDisabled
- - (void)setFd_interactivePopDisabled:(BOOL)disabled
//fd_prefersNavigationBarHidden属性的getter/setter方法
- - (BOOL)fd_prefersNavigationBarHidden
- - (void)setFd_prefersNavigationBarHidden:(BOOL)hidden
//fd_interactivePopMaxAllowedInitialDistanceToLeftEdge属性的getter/setter方法
- - (CGFloat)fd_interactivePopMaxAllowedInitialDistanceToLeftEdge
- - (void)setFd_interactivePopMaxAllowedInitialDistanceToLeftEdge:(CGFloat)distance