导航控制器默认自带了侧滑功能,当用户在界面的左边滑动的时候,就会有侧滑功能。但是如何实现全屏滑动返回效果?
分析:
第一步分析:
1.导航控制器的view自带了滑动手势,只不过手势的触发范围只能在左边。
2.当用户在界面左边拖动,就会触发滑动手势方法,并且有滑动返回功能,说明系统手势触发的方法已经实现了滑动返回功能。
3.为什么说系统手势触发的方法已经实现了滑动返回功能?
原因:
- 创建滑动手势对象的时候,需要绑定监听者,当触发手势的时候会调用target的action。
UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:action];
- 当用户在界面左边滑动,有滑动返回功能,这是因为触发手势了,调用target的action方法,说明action方法内部实现滑动返回功能,否则就不会有滑动返回效果。
第二步分析:
打印导航控制器自带的滑动手势:
NSLog(@"%@",self.interactivePopGestureRecognizer);
得 到: <UIScreenEdgePanGestureRecognizer: 0x7fe243d29150; state = Possible; delaysTouchesBegan = YES; view = <UILayoutContainerView 0x7fe243d28d90>; target= <(action=handleNavigationTransition:, target=<_UINavigationInteractiveTransition 0x7fe243d27b30>)>>
可知:
1.系统自带的手势是UIScreenEdgePanGestureRecognizer类型对象,屏幕边缘滑动手势
2.系统自带手势target是_UINavigationInteractiveTransition类型的对象
3.target调用的action方法名叫handleNavigationTransition:
UIScreenEdgePanGestureRecognizer,看名称就知道,这个手势的范围只能在屏幕的周边,就是因为这个手势,系统自带的滑动效果,只能实现侧边滑动。
第三步分析:
现在只差target, _UINavigationInteractiveTransition 真实类型
通过打印系统自带的滑动手势的代理,发现正好是_UINavigationInteractiveTransition对象,因此可猜测这个代理对象就是target对象,只要拿到它,就拿到系统自带滑动手势的target对象。
// 打印系统自带滑动手势的代理对象
NSLog(@"%@",self.interactivePopGestureRecognizer.delegate);
得到:<_UINavigationInteractiveTransition: 0x7ffabb732cb0>
实现代码:
导航控制器全屏滑动注意点:
1.禁止系统自带滑动手势使用。
2.只有导航控制器的非根控制器才需要触发手势,使用手势代理,控制手势触发。
// 获取系统自带滑动手势的target对象
target = self.interactivePopGestureRecognizer.delegate;
// 一定要记住禁止系统的手势
self.interactivePopGestureRecognizer.enabled = NO;
//添加自己的全屏滑动手势
UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];
[self.view addGestureRecognizer:pan];
// 当是根控制器的时候禁止手势, 当不是根控制器的时候打开手势
// 设置手势代理,拦截手势触发, 当手势开始滑动的时候就判断
pan.delegate = self;
}
#pragma mark -UIGestureRecognizerDelegate
//当手势开始滑动 作用:拦截手势触发
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer{
//子控制器个数只剩下一个(这一个就是根控制器),手势不可用
BOOL open = self.childViewControllers.count != 1;
return open;
}