浅探iOS手势

昨晚就开始打算  如果今天有空就研究一下 图片的放大缩小以及旋转功能是怎么实现的

在没开始之前 我的思路是:

对于放大缩小

我们先记录开始我们手指2个点  然后算距离 

在change的方法里 计算2点的距离  然后按比例缩放


对于旋转 

先获取2点  然后计算2点的角度
如图


浅探iOS手势_第1张图片

然后我们在change的方法里 也计算角度 然后和原创角度对比 再旋转相应的度数

然后对于图片的变好我是想改变frame和layer属性来实现的

但是 对于我想到的事情苹果又比我早想到了 = =


实现代码如下所示:

实现放大缩小的
- (void)handlePinch:(UIPinchGestureRecognizer *)recognizer {

//在已缩放大小基础下进行累加变化;区别于:使用 CGAffineTransformMakeScale 方法就是在原大小基础下进行变化

CGFloat scale = recognizer.scale;

//    NSLog(@"1  ==  %f",recognizer.scale);

recognizer.view.transform = CGAffineTransformScale(recognizer.view.transform, scale, scale);

recognizer.scale = 1.0; //记得设置为1

}


实现旋转的

/**

*  旋转手势

*/

- (void)handleRotation:(UIRotationGestureRecognizer *)recognizer {

recognizer.view.transform = CGAffineTransformRotate(recognizer.view.transform, recognizer.rotation);

recognizer.rotation = 0.0;//这里记得设置为0

}

创建手势的代码 如下所示:

UIPinchGestureRecognizer *pinch = [[UIPinchGestureRecognizer alloc] initWithTarget:self action:@selector(handlePinch:)];

[view addGestureRecognizer:pinch];

UIRotationGestureRecognizer *rotation = [[UIRotationGestureRecognizer alloc] initWithTarget:self action:@selector(handleRotation:)];

[view addGestureRecognizer:rotation];


好了做完自己想要的功能了

总结:1.transform 这属性可以实现view的放大缩小  旋转 的功能;

          2.记得设置 recognizer.scale = 1.0;  和  recognizer.rotation = 0.0;


轻扫手势 (不是轻扫屏幕边缘,是整个屏幕都可以)

//添加轻扫手势

UISwipeGestureRecognizer *swipeGesture = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(swipeGesture:)];

//设置轻扫的方向

swipeGesture.direction = UISwipeGestureRecognizerDirectionRight; //默认向右

[self.view addGestureRecognizer:swipeGesture];

//添加轻扫手势

UISwipeGestureRecognizer *swipeGestureLeft = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(swipeGesture:)];

//设置轻扫的方向

//    swipeGestureLeft.direction = UISwipeGestureRecognizerDirectionLeft; //默认向右

[self.view addGestureRecognizer:swipeGestureLeft];


比平常多了一个设置方向的设置


//轻扫手势触发方法

-(void)swipeGesture:(id)sender

{

UISwipeGestureRecognizer *swipe = sender;

if (swipe.direction == UISwipeGestureRecognizerDirectionLeft)

{

NSLog(@"左边");

}

if (swipe.direction == UISwipeGestureRecognizerDirectionRight)

{

NSLog(@"右边");

}

}


再把我之前写的 长按手势 的代码也贴上


/**

*  长按手势

*/

@property (nonatomic, strong) UILongPressGestureRecognizer *longPress;



_longPress = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(lonePressMoving:)];

[self.JPcollectionView addGestureRecognizer:_longPress];



- (void)lonePressMoving:(UILongPressGestureRecognizer *)longPress {

switch (_longPress.state) {

case UIGestureRecognizerStateBegan: {

{

CLog(@"开始");

}

break;

}

case UIGestureRecognizerStateChanged: {

//拖动

break;

}

case UIGestureRecognizerStateEnded: {

CLog(@"结束");

break;

}

default: [self.JPcollectionView cancelInteractiveMovement];

break;

}

}


iOS的手势有七种

UIPanGestureRecognizer(拖动)

UIPinchGestureRecognizer(捏合)

UIRotationGestureRecognizer(旋转)

UITapGestureRecognizer(点按)

UILongPressGestureRecognizer(长按)

​UISwipeGestureRecognizer(轻扫)

自定义


自定义的思路: 1.继承 UIGestureRecognizer 类 (记得导入 #import
                          2.定制下面的方法      

- (void)reset;

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event;

- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event;

- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event;

- (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event;

//以上方法在分类 UIGestureRecognizer (UIGestureRecognizerProtected) 中声明,更多方法声明请自行查看

//单击手势 (自定义的)

UITapGestureRecognizer *tapGesture = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(showPicker)];

tapGesture.numberOfTapsRequired = 1;//手势敲击的次数

[self.view addGestureRecognizer:tapGesture];

然后就写方法就可以了



UIGestureRecognizer 的继承关系如下:

浅探iOS手势_第2张图片

手势状态

在前六种手势识别中,只有一种手势是离散型手势,它就是 UITapGestureRecognizer。

离散型手势的特点就是:一旦识别就无法取消,而且只会调用一次手势操作事件(初始化手势时指定的回调方法)。

​换句话说其他五种手势是连续型手势,而连续型手势的特点就是:会多次调用手势操作事件,而且在连续手势识别后可以取消手势。从下图可以看出两者调用操作事件的次数是不同的:

浅探iOS手势_第3张图片

手势状态枚举如下:

typedef NS_ENUM(NSInteger, UIGestureRecognizerState) {

UIGestureRecognizerStatePossible,  // 尚未识别是何种手势操作(但可能已经触发了触摸事件),默认状态

UIGestureRecognizerStateBegan,      // 手势已经开始,此时已经被识别,但是这个过程中可能发生变化,手势操作尚未完成

UIGestureRecognizerStateChanged,    // 手势状态发生转变

UIGestureRecognizerStateEnded,      // 手势识别操作完成(此时已经松开手指)

UIGestureRecognizerStateCancelled,  // 手势被取消,恢复到默认状态

UIGestureRecognizerStateFailed,    // 手势识别失败,恢复到默认状态

UIGestureRecognizerStateRecognized = UIGestureRecognizerStateEnded // 手势识别完成,同UIGestureRecognizerStateEnded

};


对于离散型手势 UITapGestureRecgnizer 要么被识别,要么失败,点按(假设点按次数设置为1,并且没有添加长按手势)下去一次不松开则此时什么也不会发生,松开手指立即识别并调用操作事件,并且状态为3(已完成)。

但是连续型手势要复杂一些,就拿旋转手势来说,如果两个手指点下去不做任何操作,此时并不能识别手势(因为我们还没旋转)但是其实已经触发了触摸开始事件,此时处于状态0;如果此时旋转会被识别,也就会调用对应的操作事件,同时状态变成1(手势开始),但是状态1只有一瞬间;紧接着状态变为2(因为我们的旋转需要持续一会),并且重复调用操作事件(如果在事件中打印状态会重复打印2);松开手指,此时状态变为3,并调用1次操作事件。


下面就说一下装逼一点的了

手势穿透

适用场景ViewA 当住了部分ViewB(A在B的上面) 但是A中可以看到B的部分  然后点了AB重复的部分 B触发方法


如何实现:
重写 响应链中的   -(UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event;  方法

这个方法做了什么事情呢?
1.判断当前控件userInteractionEnabled、hidden、alpha这三个属性的值

2.调用 pointInside: withEvent: 方法

3.从后向前遍历子控件,并调用子控件的 hitTest: withEvent: 和 pointInside: withEvent: 方法

因为是不开源的 所以猜测的代码如下

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {

    if (self.userInteractionEnabled == NO || self.hidden == YES || self.alpha <= 0.01) {

        return nil;

    }

    if ([self pointInside:point withEvent:event] == NO) {

        return nil;

    }

    int count = (int)self.subviews.count - 1;

    for (int i = count; i >= 0 ; i--) {

        UIView *view = self.subviews[i];

        CGPoint p = [view convertPoint:point fromView:self];

        if ([view pointInside:p withEvent:event]) {

            return [view hitTest:p withEvent:event];

            break;

         }

}

return self;

}


- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent*)event;

这个方法的作用就是判断这个点 它在不在他的范围内  它能不能响应这个点击事件


若和实现手势穿透呢

判断当前的view是不是我大的那个view  如果是我就让这方法继续走  如果不是就返回nil


用黑科技实现手势穿透


要是情况允许 我们可以设置tag值  用tag值 和point 来实现 我们想要的效果





你可能感兴趣的:(浅探iOS手势)