前言
此次文章,讲述的是导航控制器全屏滑动返回效果,而且代码量非常少,10行内搞定。
效果如图:
效果.gif
一、自定义导航控制器
目的:以后需要使用全屏滑动返回功能,就使用自己定义的导航控制器。
二、分析导航控制器侧滑功能
效果:导航控制器默认自带了侧滑功能,当用户在界面的左边滑动的时候,就会有侧滑功能。
系统自带的侧滑效果:
侧滑.gif分析:
1.导航控制器的view自带了滑动手势,只不过手势的触发范围只能在左边。
2.当用户在界面左边拖动,就会触发滑动手势方法,并且有滑动返回功能,说明系统手势触发的方法已经实现了滑动返回功能。
3.为什么说系统手势触发的方法已经实现了滑动返回功能?
原因:
- 创建滑动手势对象的时候,需要绑定监听者,当触发手势的时候会调用target的action。
UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:action];
- 当用户在界面左边滑动,有滑动返回功能,这是因为触发手势了,调用target的action方法,
说明action方法内部实现滑动返回功能
,否则就不会有滑动返回效果。
三、实现全屏滑动功能分析
- 打印导航控制器自带的滑动手势,看下它的真实面目。
- 系统自带的滑动手势
interactivePopGestureRecognizer
// self指向的导航控制器,在导航控制器的viewDidLoad方法打印
- (void)viewDidLoad {
[super viewDidLoad];
NSLog(@"%@",self.interactivePopGestureRecognizer);
}
打印结果图片:
Snip20150801_1.png- 由图中可知:
- 1.系统自带的手势是
UIScreenEdgePanGestureRecognizer
类型对象,屏幕边缘滑动手势 - 3.系统自带手势target是
_UINavigationInteractiveTransition
类型的对象 - 4.target调用的action方法名叫
handleNavigationTransition:
分析:
-
UIScreenEdgePanGestureRecognizer
,看名称就知道,这个手势的范围只能在屏幕的周边,就是因为这个手势,系统自带的滑动效果,只能实现侧边滑动。
四、如何实现全屏滑动功能
-
给自己的导航控制器,添加一个
全屏的滑动手势
,调用系统自带滑动手势的target
的action
方法,利用系统实现的滑动返回功能,
加上自己全屏滑动手势
,就有全屏滑动功能
了。 -
问题:
如何拿到系统自带的target对象?
,action方法名已经知道,而且系统肯定在target对象实现了,只要拿到target对象,调用这个方法就行。 -
通过打印系统自带的滑动手势的代理,发现正好是
_UINavigationInteractiveTransition对象
,因此我猜测这个代理对象就是target对象,只要拿到它,就拿到系统自带滑动手势的target对象。
// 打印系统自带滑动手势的代理对象
NSLog(@"%@",self.interactivePopGestureRecognizer.delegate);
- 打印图片:
-
导航控制器全屏滑动注意点:
-
1.
禁止系统自带滑动手势使用
。 -
2.
只有导航控制器的非根控制器才需要触发手势
,使用手势代理,控制手势触发。 -
全屏滑动代码实现
- (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;
}
联系方式
如果你喜欢这篇文章,可以继续关注我,微博:吖了个峥,欢迎交流。
网友评论
id target = self.navigationController.interactivePopGestureRecognizer.delegate;
UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];
[self.view addGestureRecognizer:pan];
导航控制器中除根控制器外的所有视图控制器都能实现侧滑,而且不需要只滑动边缘
首先,这个检查一定是个静态检查吧,不可能是运行时检查,因为代码逻辑那么复杂,把程序跑起来看所有 objc_msgSend 中包不包括私有调用这件事太不现实了。
对 ipa 文件做静态检查的话肯定是去分析 Mach-O 可执行文件,因为这时很多源代码级别的信息已经丢失,经分析可以采取下面几种手段:
是否 link 了私有 framework 或者公开 framework 中的私有符号,这可以防止开发者把私有 header 都 dump 出来供程序直接调用。
同上,使用@selector(_private_sel)加上-performSelector:的方式直接调用私有 API。
扫描所有符号,查看是否有继承自私有类,重载私有方法,方法名是否有重合。
扫描所有string,看字符串常量段是否出现和私有 API 对应的。
我觉得前三条被 catch 住的可能性最高,也最容易被检查出来。再来看我们用到用字符串的方法 kvc 和 反射 selector,应该属于最后一条,这时候就很难抉择了,拿 handleNavigationTransition: 来说,看上去人畜无害啊,我自己类里面的方法也完全可能命名出这个来,所以单单凭借字符串命中私有 API 判定,苹果很容易误伤一大票开发者。
综上,我觉得使用字符串的方式使用私有 API 是相对安全的,我们的 App 马上要提交审核,如果过了几天你还能读到这段文字,说明我的猜想是木有错的,大家可以放心使用。"
审核通过了,可以看下这篇文章.http://blog.sunnyxx.com/2015/06/07/fullscreen-pop-gesture/