1. UIPanGestureRecognizer(拖动)基本介绍
- 父类是UIGestureRecognizer
2. UIPanGestureRecognizer应用
// 拖拽
- (void)setUpPan
{
// 拖拽
UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(pan:)];
[self.imageView addGestureRecognizer:pan];
}
- (void)pan:(UIPanGestureRecognizer *)pan
{
//获取偏移量
// 返回的是相对于最原始的手指的偏移量
CGPoint transP = [pan translationInView:self.imageView];
// 移动图片控件
self.imageView.transform = CGAffineTransformTranslate(self.imageView.transform, transP.x, transP.y);
// 复位,表示相对上一次
[pan setTranslation:CGPointZero inView:self.imageView];
}
3. 右滑返回手势
3.1 全局返回手势
-
系统只能左侧滑动,查看系统实现的手势类型,对象及方法,分析后发现手势类型是屏幕边缘滑动手势,实现对象就是手势代理,方法是handleNavigationTransition;
-
如何查看:打印系统侧滑手势,可知道手势类型与方法;通过断点打印方式获取系统手势的target,最后发现用kvc获取self.interactivePopGestureRecognizer的@“_targets”得出targets数组,再通过kvc获取targets[0]的@“_target”属性;或者直接获取代理;
-
实现方案:创建一个全屏手势,调用系统的滑动返回功能,添加到非根控制器的view上,设置系统自带的手势不可用。
// 禁止使用系统自带的手势
self.interactivePopGestureRecognizer.enabled = NO;
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wundeclared-selector" //取消方法警告
//#pragma clang diagnostic ignored "-Wdeprecated-declarations"//取消声明警告
// 全局手势
UIPanGestureRecognizer *popPan = [[UIPanGestureRecognizer alloc]initWithTarget:self.interactivePopGestureRecognizer.delegate action:@selector(handleNavigationTransition:)];
#pragma clang diagnostic pop
// 给控制器的view添加全局返回手势
[viewController.view addGestureRecognizer:popPan];
// 手势与手势代理是在屏幕完全显示时才设置
// 程序创建时,加载所有导航控制器并没有完全显示屏幕,所以没有设置手势及手势代理
}
3. 2 边缘侧滑返回手势
-
导航控制器跳转后,覆盖返回按钮,不会触发侧滑返回手势,若需要边缘侧滑返回手势:
-
思路:让手势代理判断手势方法-边缘滑动返回方法是否实现,若是根控制器不实现;非控制器则允许触发手势方法。边缘滑动返回方法已经由系统实现,我们只需管理其是否触发尽可
-
方法一:让导航控制器成为其手势代理,实现代理方法gestureRecognizerShouldBegin,即可;但要根据子控制器数量返回Bool值,否则会假死。手势代理方法详见“UIGestureRecognizer”
self.interactivePopGestureRecognizer.delegate = self;
// 判断下是否允许触发当前手势
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer
{
// 如果不是根控制器,就触发手势
return self.childViewControllers.count > 1;
}
- 方法二:详见下面代码(不推荐,仅用于理解全局手势底层)
-(void)pushViewController:(UIViewController *)viewController animated:(BOOL)animated
{
//判断是不是导航控制器的根控制器
if (self.childViewControllers.count != 0) {
//设置导航条左边按钮
viewController.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc]initWithImage:[UIImage originalImageNamed:@"NavBack"] style:UIBarButtonItemStyleDone target:self action:@selector(back)];
/** 跳转时隐藏标签栏 */
viewController.hidesBottomBarWhenPushed = YES;
//获取并记录手势代理
self.popDelegate = self.interactivePopGestureRecognizer.delegate;
//清空原手势代理,gestureRecognizerShouldBegin默认返回YES,手势方法-边缘侧滑返回可以执行
self.interactivePopGestureRecognizer.delegate = nil;
//设置代理
self.delegate = self;
//执行系统的跳转方法
[super pushViewController:viewController animated:animated];
}
//屏幕完全显示时才设置手势代理,屏幕完全显示这个方法会调用两次
-(void)navigationController:(UINavigationController *)navigationController didShowViewController:(UIViewController *)viewController animated:(BOOL)animated
{
//判断是不是根控制器
if (self.childViewControllers.count == 1) {
// 恢复手势代理 - 手势代理底层会判断是否是根控制器,若是根控制器,gestureRecognizerShouldBegin返回NO,不执行手势方法
// 在不覆盖返回按钮时,手势代理会让上面逻辑执行,但覆盖后,无论是否是根控制器,手势代理方法gestureRecognizerShouldBegin都返回NO,不执行手势方法
self.interactivePopGestureRecognizer.delegate = self.popDelegate;
}
4. UIPanGestureRecognizer属性
// 要求的最小触摸数量,默认是1
@property (nonatomic) NSUInteger minimumNumberOfTouches __TVOS_PROHIBITED;
// 允许的最大触摸数量,默认是无穷大
@property (nonatomic) NSUInteger maximumNumberOfTouches __TVOS_PROHIBITED;
5. UIPanGestureRecognizer方法
// 位移
- (CGPoint)translationInView:(nullable UIView *)view;
// 设置位移
- (void)setTranslation:(CGPoint)translation inView:(nullable UIView *)view;
// 位移速度
- (CGPoint)velocityInView:(nullable UIView *)view;
网友评论