我们都知道,iOS7导航控制器默认自带了侧滑功能,当用户在界面的左边滑动的时候,就会有侧滑功能。但是如果我们从从导航控制器的返回按钮,就发现系统所带的侧滑返回功能无法使用。因此为了解决此问题,有以下方法实现:
方法一:导航控制器全屏滑动返回效果
当用户在界面左边拖动,就会触发滑动手势方法,并且有滑动返回功能,说明系统手势触发了方法,即调用了target的action方法,也就是说action方法内已经实现侧滑返回。 系统自带的滑动手势interactivePopGestureRecognizer
// self指向的导航控制器,在导航控制器的viewDidLoad方法打印
- (void)viewDidLoad {
[super viewDidLoad];
NSLog(@"%@",self.interactivePopGestureRecognizer);
}
打印结果知:
1.系统自带的手势是UIScreenEdgePanGestureRecognizer类型对象,屏幕边缘滑动手势
2.系统自带手势target是_UINavigationInteractiveTransition类型的对象
3.target调用的action方法名叫handleNavigationTransition:
全屏滑动代码块实现
- (void)viewDidLoad {
[super viewDidLoad];
// 获取系统自带滑动手势的target对象
id target = self.interactivePopGestureRecognizer.delegate;
// 创建全屏滑动手势,调用系统自带滑动手势的target的action方法
UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];
// 设置手势代理,拦截手势触发 pan.delegate = self;
// 给导航控制器的view添加全屏滑动手势
[self.view addGestureRecognizer:pan];
// 禁止使用系统自带的滑动手势
self.interactivePopGestureRecognizer.enabled = NO;
}
// 什么时候调用:每次触发手势之前都会询问下代理,是否触发。
// 作用:拦截手势触发
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer{
// 注意:只有非根控制器才有滑动返回功能,根控制器没有。
// 判断导航控制器是否只有一个子控制器,如果只有一个子控制器,肯定是根控制器
if (self.childViewControllers.count == 1) {
// 表示用户在根控制器界面,就不需要触发滑动手势,
return NO; }
return YES;
}
导航控制器全屏滑动注意点:
- 1.禁止系统自带滑动手势使用。
- 2.只有导航控制器的非根控制器才需要触发手势,使用手势代理,控制手势触发。
完成、这样就搞定了, 亲测此方法会与界面有滑动手势的产生冲突,因此有第二种方法!
以上方法参考原文链接如下:
http://www.cocoachina.com/ios/20150811/12897.html?utm_source=tuicool)
方法二:实现自定义导航控制器边缘滑动返回
方法二的实现原理和方法一一样,只不过它还是使用的系统的边缘手势实现侧滑返回功能。只需要在每个类里面添加如下代码块:
写在.m中, 别忘了遵守 UIGestureRecognizerDelegate协议。
//当前导航控制器@property (nonatomic, strong) UIViewController *currentShowVC;
-(void)viewWillAppear:(BOOL)animated {
//设置代理
self.navigationController.interactivePopGestureRecognizer.delegate =(id)self;
//启用系统自带的滑动手势
self.navigationController.interactivePopGestureRecognizer.enabled = YES;
//判断导航控制器是否只有一个子控制器,如果只有一个子控制器,肯定是根控制器,这里我的项目是有tabbar,所以在页面切换之间设置是否显示tababr。
if (self.navigationController.viewControllers.count == 1){
//将当前导航控制器置空
self.currentShowVC = Nil;
[SharedAppDelegate setTabBarHidden:NO animated:YES];
}else{ 如果不是根控制器,就设置当前导航控制器为其本身。
self.currentShowVC = self;
[SharedAppDelegate setTabBarHidden:YES animated:NO];
}
}
手势的代理方法
-(BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer{
if (gestureRecognizer ==
self.navigationController.interactivePopGestureRecognizer) { //当前导航控制器是根视图控制器
//the most important
return (self.currentShowVC ==
self.navigationController.topViewController);
// 不要隐藏tabbar
[SharedAppDelegate setTabBarHidden:NO animated:NO];
}
return YES;
}
方法三:(推荐使用)
- 自定义UINavigationController
#import "ZGKNavigationViewController.h"
@interface ZGKNavigationViewController ()<UIGestureRecognizerDelegate>
@end
@implementation ZGKNavigationViewController
- (void)viewDidLoad {
[super viewDidLoad];
// self.interactivePopGestureRecognizer.delegate = nil;就是什么时候都可以滑动手势,会有bug,无法点击settingBtn
// 设置手势代理
self.interactivePopGestureRecognizer.delegate = self;
// self.interactivePopGestureRecognizer.enabled = YES;
// 设置导航控制器navigationBar和tabBar的背景颜色可以解决导航栏黑色点的bug
[self.navigationBar setBackgroundImage:[UIImage imageNamed:@"navigationbarBackgroundWhite"] forBarMetrics:UIBarMetricsDefault];
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
/**
* 重写push方法的目的 : 拦截所有push进来的子控制器(包括代码和storyBoard实现的)
*
* @param viewController 刚刚push进来的子控制器
*/
- (void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated{
if (self.childViewControllers.count > 0) { // 如果viewController不是最早push进来的子控制器,则不用设置返回按钮
// 左上角
UIButton *backButton = [UIButton buttonWithType:UIButtonTypeCustom];
[backButton setImage:[UIImage imageNamed:@"navigationButtonReturn"] forState:UIControlStateNormal];
[backButton setImage:[UIImage imageNamed:@"navigationButtonReturnClick"] forState:UIControlStateHighlighted];
[backButton setTitle:@"返回" forState:UIControlStateNormal];
[backButton setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
[backButton setTitleColor:[UIColor redColor] forState:UIControlStateHighlighted];
[backButton sizeToFit];
// 这句代码放在sizeToFit后面
backButton.contentEdgeInsets = UIEdgeInsetsMake(0, -20, 0, 0);
[backButton addTarget:self action:@selector(back) forControlEvents:UIControlEventTouchUpInside];
viewController.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithCustomView:backButton];
// 隐藏底部的工具条
viewController.hidesBottomBarWhenPushed = YES;
}
// 所有设置搞定后, 再push控制器(不要忘记调用父类方法)
[super pushViewController:viewController animated:animated];
}
- (void)back{
[self popViewControllerAnimated:YES];
}
//- (UIViewController *)popViewControllerAnimated:(BOOL)animated
//{
// return [super popViewControllerAnimated:NO];
//}
//
//- (NSArray<UIViewController *> *)popToViewController:(UIViewController *)viewController animated:(BOOL)animated
//{
// return [super popToViewController:viewController animated:NO];
//}
//
//- (NSArray<UIViewController *> *)popToRootViewControllerAnimated:(BOOL)animated
//{
// return [super popToRootViewControllerAnimated:NO];
//}
#pragma mark - <UIGestureRecognizerDelegate>
/**
* 手势识别器对象会调用这个代理方法来决定手势是否有效
*
* @return YES : 手势有效, NO : 手势无效
*/
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
{
// if (self.childViewControllers.count == 1) {
// return NO;
// }
//
// return YES;
// 手势何时有效 : 当导航控制器的子控制器个数 > 1就有效
return self.childViewControllers.count > 1;
}
@end