只要继承了UIResponder的对象就可作为事件的响应者,实际上UIControl继承了UIView,UIView又继承了UIResponder,由此可见,所有的对象都可作为事件的响应者。
当用户与某个控件交互时,该控件将作为“第一响应者(First Responder)”,第一响应者将作为响应者链的开始,该事件首先被发送给第一响应者(也就是用户触摸屏幕的控件)。事件将沿着响应者链一直向下传播,直到被某个响应者处理。事件响应者链的典型传播路线如下:
First Responder —> First Responder的视图控制器(如果有)—>父容器(如果有)—>父容器的视图控制器(如果有)—>UIWindow—>UIApplication—>应用程序委托对象
如果某个响应者“截获”了某个事件,那么该响应者要根据条件决定是否处理该事件。当响应者无法处理该事件时,则需要在处理方法中手动传递该事件。例如如下代码:
- (void)handleTapEvent:(UITapGestureRecognizer *)event{
if (YES) {
//处理事件
}else{
OtherTapView* view = (OtherTapView *)self.nextResponder;
[view handleTapEvent:event];
}
}
如果希望自定义控件可以响应用户的触碰事件,则可以通过通过UIResponder的如下四个方法实现:
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
NSLog(@"%ld",[touches count]);
NSLog(@"%ld",[[touches anyObject]tapCount]);
CGPoint point = [[touches anyObject] locationInView:self.view];
}
注意:
UIGestureRecognizer提供了如下子类:
使用手势处理器处理用户触碰手势的编程步骤如下:
UIGestureRecognizer作为所有手势处理器的基类,它提供了如下常用的方法和属性:
typedef NS_ENUM(NSInteger, UIGestureRecognizerState) {
UIGestureRecognizerStatePossible,
UIGestureRecognizerStateBegan,
UIGestureRecognizerStateChanged,
UIGestureRecognizerStateEnded,
UIGestureRecognizerStateCancelled,
UIGestureRecognizerStateFailed,
UIGestureRecognizerStateRecognized = UIGestureRecognizerStateEnded
};
UITapGestureRecognizer还提供了如下两个属性:
实例:
- (void)viewDidLoad {
[super viewDidLoad];
gv = [[UIView alloc]initWithFrame:CGRectMake(10, 200, self.view.frame.size.width - 20, 200)];
gv.layer.borderWidth = 2;
gv.layer.cornerRadius = 6;
//设置gv控件支持用户交互
gv.userInteractionEnabled = YES;
//设置gv控件支持多点触碰
gv.multipleTouchEnabled = YES;
[self.view addSubview:gv];
label = [[UILabel alloc]initWithFrame:CGRectMake(20, 50, gv.frame.size.width - 40, 40)];
[gv addSubview:label];
self.view.backgroundColor = [UIColor lightGrayColor];
for (int i = 0; i < 6; i ++) {
//创建手势处理器,指定使用该控制器的handleTap:方法处理手势
UITapGestureRecognizer* gesture = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(handleTap:)];
//设置该点击手势处理器只处理i次连击事件
gesture.numberOfTapsRequired = i;
//设置该点击手势处理器只处理两个手指的触碰事件
gesture.numberOfTouchesRequired = 2;
//为gv控件添加手势处理器
[gv addGestureRecognizer:gesture];
}
}
- (void)handleTap:(UITapGestureRecognizer *)gesture{
NSInteger touchNum = gesture.numberOfTouches;
NSUInteger tapNum = gesture.numberOfTapsRequired;
label.text = [NSString stringWithFormat:@"用户使用%ld个手指进行触碰,触碰次数为:%ld",touchNum,tapNum];
//指定2秒后清除label的文本
[label performSelector:@selector(setText:) withObject:@"" afterDelay:5];
}
UIPinchGestureRecognizer定义了如下两个属性来获取捏合相关信息:
实例:
- (void)viewDidLoad {
[super viewDidLoad];
gv = [[UIView alloc]initWithFrame:CGRectMake(10, 200, self.view.frame.size.width - 20, 200)];
gv.layer.borderWidth = 2;
gv.layer.cornerRadius = 6;
//设置gv控件支持用户交互
gv.userInteractionEnabled = YES;
//设置gv控件支持多点触碰
gv.multipleTouchEnabled = YES;
[self.view addSubview:gv];
label = [[UILabel alloc]initWithFrame:CGRectMake(20, 50, gv.frame.size.width - 40, 80)];
label.numberOfLines = 2;
[gv addSubview:label];
self.view.backgroundColor = [UIColor lightGrayColor];
UIPinchGestureRecognizer* gesture = [[UIPinchGestureRecognizer alloc]initWithTarget:self action:@selector(handlePinch:)];
//为gv控件添加手势处理器
[gv addGestureRecognizer:gesture];
}
- (void)handlePinch:(UIPinchGestureRecognizer *)gesture{
//获取用户捏合的速度和比例
CGFloat velocity = gesture.velocity;
CGFloat scale = gesture.scale;
label.text = [NSString stringWithFormat:@"用户捏合的速度为:%g,比例为:%g",velocity,scale];
//指定2秒后清除label的文本
[label performSelector:@selector(setText:) withObject:@"" afterDelay:5];
}
- (void)viewDidLoad {
[super viewDidLoad];
[UIApplication sharedApplication].statusBarHidden = YES;
srcImage = [UIImage imageNamed:@"妞.png"];
//设置图片直接显示在中间(不进行任何缩放)
self.view.contentMode = UIViewContentModeCenter;
self.view.backgroundColor = [UIColor lightGrayColor];
gv = [[UIImageView alloc]initWithFrame:CGRectMake(10, 200, self.view.frame.size.width - 20, 200)];
currentSize = CGSizeMake(self.view.frame.size.width - 20, 200);
gv.layer.borderWidth = 2;
gv.layer.cornerRadius = 6;
//设置gv控件支持用户交互
gv.userInteractionEnabled = YES;
//设置gv控件支持多点触碰
gv.multipleTouchEnabled = YES;
[self.view addSubview:gv];
gv.image = srcImage;
//创建UIPinchGestureRecognizer手势处理器,该手势处理器激发scaleImage:方法
UIPinchGestureRecognizer* gesture = [[UIPinchGestureRecognizer alloc]initWithTarget:self action:@selector(scaleImage:)];
[gv addGestureRecognizer:gesture];
}
//imageView进行缩放
- (void)scaleImage:(UIPinchGestureRecognizer *)gesture{
gv.bounds = CGRectMake(0, 0, currentSize.width * gesture.scale, currentSize.height * gesture.scale);
if (gesture.state == UIGestureRecognizerStateEnded) {
currentSize = CGSizeMake(currentSize.width * gesture.scale, currentSize.height * gesture.scale);
}
}
UIRotationGestureRecognizer定义了如下两个属性来获取旋转相关信息:
实例:通过旋转手势旋转图片
- (void)viewDidLoad {
[super viewDidLoad];
srcImage = [UIImage imageNamed:@"妞.png"];
//设置图片直接显示在中间(不进行任何缩放)
self.view.backgroundColor = [UIColor lightGrayColor];
gv = [[UIImageView alloc]initWithFrame:CGRectMake(10, 200, self.view.frame.size.width - 20, 200)];
currentSize = CGSizeMake(self.view.frame.size.width - 20, 200);
gv.layer.borderWidth = 2;
gv.layer.cornerRadius = 6;
//设置gv控件支持用户交互
gv.userInteractionEnabled = YES;
//设置gv控件支持多点触碰
gv.multipleTouchEnabled = YES;
[self.view addSubview:gv];
gv.image = srcImage;
//创建UIPinchGestureRecognizer手势处理器,该手势处理器激发scaleImage:方法
UIRotationGestureRecognizer* gesture = [[UIRotationGestureRecognizer alloc]initWithTarget:self action:@selector(rotateImage:)];
[gv addGestureRecognizer:gesture];
}
- (void)rotateImage:(UIRotationGestureRecognizer *)gesture{
//根据当前缩放的比例计算缩放后的目标图片大小
gv.transform = CGAffineTransformMakeRotation(imageRotate + gesture.rotation);
if (gesture.state == UIGestureRecognizerStateEnded) {
imageRotate += gesture.rotation;
}
}
UISwipeGestureRecognizer定义了如下两个属性来设置相关信息:
typedef NS_OPTIONS(NSUInteger, UISwipeGestureRecognizerDirection) {
UISwipeGestureRecognizerDirectionRight = 1 << 0,
UISwipeGestureRecognizerDirectionLeft = 1 << 1,
UISwipeGestureRecognizerDirectionUp = 1 << 2,
UISwipeGestureRecognizerDirectionDown = 1 << 3
};
实例:
- (void)viewDidLoad {
[super viewDidLoad];
//设置图片直接显示在中间(不进行任何缩放)
self.view.backgroundColor = [UIColor lightGrayColor];
gv = [[UIImageView alloc]initWithFrame:CGRectMake(10, 200, self.view.frame.size.width - 20, 200)];
gv.layer.borderWidth = 2;
gv.layer.cornerRadius = 6;
//设置gv控件支持用户交互
gv.userInteractionEnabled = YES;
//设置gv控件支持多点触碰
gv.multipleTouchEnabled = YES;
[self.view addSubview:gv];
label = [[UILabel alloc]initWithFrame:CGRectMake(20, 50, gv.frame.size.width - 40, 80)];
label.numberOfLines = 2;
[gv addSubview:label];
for (int i = 0 ; i < 4; i ++) {
//创建手势处理器,指定使用该控制器的handleSwipe:方法处理轻扫手势
UISwipeGestureRecognizer* gesture = [[UISwipeGestureRecognizer alloc]initWithTarget:self action:@selector(handleSwipe:)];
//设置该手势处理器只处理i个手指的轻扫手势
gesture.numberOfTouchesRequired = 1;
//指定该手势处理器只处理1 << i方向的轻扫手势
gesture.direction = 1 << i;
[gv addGestureRecognizer:gesture];
}
}
- (void)handleSwipe:(UISwipeGestureRecognizer *)gesture{
//获取手势的方向
NSUInteger direction = gesture.direction;
//根据手势方向的值得到方向字符串
NSString* dirStr = direction == UISwipeGestureRecognizerDirectionRight?@"向右":(direction == UISwipeGestureRecognizerDirectionLeft?@"向左":(direction == UISwipeGestureRecognizerDirectionUp?@"向上":@"向下"));
NSLog(@"dirStr = %@",dirStr);
NSUInteger touchNum = gesture.numberOfTouchesRequired;
label.text = [NSString stringWithFormat:@"用户使用%ld个手指进行轻扫,方向为%@",touchNum,dirStr];
//指定5秒后清除label的文本
[label performSelector:@selector(setText:) withObject:@"" afterDelay:5];
}
UIPanGestureRecognizer定义了如下的方法来获取拖动相关信息:
例如:
- (void)viewDidLoad {
[super viewDidLoad];
//设置图片直接显示在中间(不进行任何缩放)
self.view.backgroundColor = [UIColor lightGrayColor];
gv = [[UIImageView alloc]initWithFrame:CGRectMake(10, 200, self.view.frame.size.width - 20, 200)];
gv.layer.borderWidth = 2;
gv.layer.cornerRadius = 6;
//设置gv控件支持用户交互
gv.userInteractionEnabled = YES;
//设置gv控件支持多点触碰
gv.multipleTouchEnabled = YES;
[self.view addSubview:gv];
label = [[UILabel alloc]initWithFrame:CGRectMake(20, 50, gv.frame.size.width - 40, 80)];
label.numberOfLines = 3;
[gv addSubview:label];
//创建手势处理器,指定使用该控制器的handlePan:方法处理手势
UIPanGestureRecognizer* gesture = [[UIPanGestureRecognizer alloc]initWithTarget:self action:@selector(handlePan:)];
//设置该手势处理器最少需要1个手指
gesture.minimumNumberOfTouches = 1;
//设置该手势处理器最多需要2个手指
gesture.maximumNumberOfTouches = 2;
[gv addGestureRecognizer:gesture];
}
//实现手势处理器的方法,该方法应该生命一个形参
//当该方法被激发时,手势处理器会作为参数传给该方法的参数
- (void)handlePan:(UIPanGestureRecognizer *)gesture{
CGPoint velocity = [gesture velocityInView:gv];
CGPoint translation = [gesture translationInView:gv];
label.text = [NSString stringWithFormat:@"水平速度为:%g,垂直速度为:%g,水平距离为:%g,垂直距离为:%g",velocity.x,velocity.y,translation.x,translation.y];
}
UILongGestureRecognizer定义了如下属性来设置该手势处理器的相关信息:
实例:
- (void)viewDidLoad {
[super viewDidLoad];
self.view.backgroundColor = [UIColor lightGrayColor];
//创建一个NSMutableArray集合 用于保存过个按钮
bnArray = [[NSMutableArray alloc]init];
//创建一个手势处理器 用于检测.处理长按手势
UILongPressGestureRecognizer* gesture = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(longPress:)];
//为该控件添加手势处理器
[self.view addGestureRecognizer:gesture];
}
- (void)longPress:(UILongPressGestureRecognizer *)gesture{
//创建一个按钮
UIButton* bn = [UIButton buttonWithType:UIButtonTypeRoundedRect];
//获取NSArray中已经包含了几个按钮
NSInteger count = bnArray.count;
//计算当前添加的按钮位于第几行第几列
NSInteger row = count/3;
NSInteger col = count%3;
//为按钮设置文本
[bn setTitle:[NSString stringWithFormat:@"按钮%ld",bnIndex] forState:UIControlStateNormal];
//设置该bn按钮的大小和位置
bn.frame = CGRectMake(MARGINE + col * CELL_WIDTH, row * CELL_HEIGHT + MARGINE, BUTTON_WIDTH, BUTTON_HEIHGT);
//为该按钮添加事件处理方法
[bn addTarget:self action:@selector(remove:) forControlEvents:UIControlEventTouchUpInside];
[bnArray addObject:bn];
//将按钮添加到应用界面的UIView控件中
[self.view addSubview:bn];
bnIndex ++;
}
- (void)remove:(id)sender{
//删除事件源控件(激发该事件的按钮)
[sender removeFromSuperview];
//将触发该事件的按钮从NSMutableArray集合中删除
[bnArray removeObject:sender];
[self rearrange];
}
- (void)rearrange{
//重新计算每个按钮的大小和位置
for (int i = 0; i < bnArray.count ; i ++) {
NSInteger row = i/3;
NSInteger col = i%3;
UIButton* bn = [bnArray objectAtIndex:i];
bn.frame = CGRectMake(MARGINE + col * CELL_WIDTH, row * CELL_HEIGHT + MARGINE, BUTTON_WIDTH, BUTTON_HEIHGT);
}
}
开发自定义手势处理器的步骤比较简单,按如下步骤操作即可:
①、创建继承UIGestureRecognizer的子类;
②、重写UIGestureRecognizer基类的触碰相关的4个方法.通过在这些方法中识别用户手指划过的痕迹—当用户手指划过的痕迹符合手势要求时,程序将该手势state设为UIGestureRecognizerStateEnaded即可.
待定
手势识别是具有互斥的原则的,比如单击和双击,如果它识别出一种手势,其后的手势将不被识别。所以对于关联手势,要做特殊处理以帮助程序甄别,应该把当前手势归结到哪一类手势里面。
比如,单击和双击并存时,如果不做处理,它就只能发送出单击的消息。为了能够识别出双击手势,就需要做一个特殊处理逻辑,即先判断手势是否是双击,在双击失效的情况下作为单击手势处理。使用
[A requireGestureRecognizerToFail:B]函数,它可以指定当A手势发生时,即便A已经滿足条件了,也不会立刻触发,会等到指定的手势B确定失败之后才触发。
- (void)requireGestureRecognizerToFail:(UIGestureRecognizer *)otherGestureRecognizer;
//例如
[singleRecognizer requireGestureRecognizerToFail:doubleRecognizer];
1、
// called when a gesture recognizer attempts to transition out of UIGestureRecognizerStatePossible. returning NO causes it to transition to UIGestureRecognizerStateFailed
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer *)gestureRecognizer;
手势可能发生的条件,比如某些特殊情况下,不想让此手势发生,就return NO。
2、
// called when the recognition of one of gestureRecognizer or otherGestureRecognizer would be blocked by the other
// return YES to allow both to recognize simultaneously. the default implementation returns NO (by default no two gestures can be recognized simultaneously)
//
// note: returning YES is guaranteed to allow simultaneous recognition. returning NO is not guaranteed to prevent simultaneous recognition, as the other gesture's delegate may return YES
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer;
gestureRecognizer可能被otherGestureRecognizer手势识别引起阻塞,返回YES为允许同时识别。默认返回NO(默认情况下同一种手势只能识别一个)
3、
// called before touchesBegan:withEvent: is called on the gesture recognizer for a new touch. return NO to prevent the gesture recognizer from seeing this touch
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch;
方法touchesBegan:withEvent:执行前将调用该方法,返回YES执行手势事件;返回NO,将不响应手势事件。