美文网首页iOSIOS开发
iOS手势-UIPanGestureRecognizer

iOS手势-UIPanGestureRecognizer

作者: 翻这个墙 | 来源:发表于2017-11-17 15:47 被阅读437次

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;

相关文章

网友评论

    本文标题:iOS手势-UIPanGestureRecognizer

    本文链接:https://www.haomeiwen.com/subject/kgyjvxtx.html