iOS事件传递与响应者链

用户以多种方式操纵他们的iOS设备,例如触摸屏幕或摇动设备。 iOS会解释用户何时以及如何操作硬件并将此信息传递到您的应用程序。 您的应用程序以自然和直观的方式响应操作的次数越多,对用户而言越有吸引力的体验。

一、事件分类

事件是发送到应用程序用于通知用户操作的对象。 在iOS中,事件可以采取多种形式:多点触摸事件,运动事件和用于控制多媒体的事件。 这最后一种类型的事件被称为遥控事件或者远程控制事件,因为它可以源自外部附件。而在我们开发过程中最常用的就是多点触摸事件。

Event in iOS

二、事件传递与响应链

当您设计应用程式时,可能需要动态响应事件。 例如,触摸可以发生在屏幕上的许多不同对象中,并且您必须决定您想要那个对象响应事件,并且理解该对象如何接收该事件。

当用户生成的事件发生时,UIKit创建一个包含处理事件所需信息的事件对象。 然后它将事件对象放置在活动应用程序的事件队列中。 对于触摸事件,该对象是在UIEvent对象中打包的一组触摸。 对于运动事件,事件对象因您使用的框架和您感兴趣的运动事件类型而异。

事件沿着特定路径传递,直到它被传递到可以处理它的对象。 首先,单例UIApplication对象从队列的顶部获取一个事件并分发处理。 通常,它将事件发送到应用程序的key window对象,该对象将事件传递到初始对象(initial object)进行处理。 初始对象取决于事件的类型。

  • 触摸事件:对于触摸事件,窗口对象首先尝试将事件传递到发生触摸的视图。 该视图称为命中测试视图(hit-test view)。 找到命中测试视图(hit-test view)的过程称为命中测试(hit-testing),这在Hit-Testing返回触摸发生的视图中描述。

  • 运动和遥控事件:对于这些事件,窗口对象将摇动或远程控制事件发送到第一响应者以进行处理。 第一响应者在响应者链由响应者对象组成中描述。

这些事件路径的最终目标是找到一个可以处理和响应事件的对象。 因此,UIKit首先将事件发送到最适合处理事件的对象。 对于触摸事件,该对象是命中测试视图(hit-test view),对于其他事件,该对象是第一个响应者。 以下部分更详细地说明命中测试视图(hit-test view)和第一响应者对象是如何确定的。

1. Hit-Testing返回触摸发生的视图

iOS使用命中测试(hit-testing)来查找被触摸的视图。 命中测试(hit-testing)涉及检查触摸是否在所有相关视图对象的边界内。 如果是,它会递归检查视图的所有子视图。视图层级中包含触摸点的最低的视图成为命中测试视图(hit-test view) 。 iOS确定命中测试视图(hit-test view)后,它会将触摸事件传递到该视图进行处理。

举例说明,假设用户触摸下图中的View E。 iOS通过按照此顺序检查子视图来查找命中测试视图(hit-test view):

  1. 触摸在View A的边界内,因此它检查子视图View B和View C.

  2. 触摸不在View B的界限内,但它在View C的界限内,因此它检查子视图View D和View E.

  3. 触摸不在View D的界限内,但它在View E的界限内。

    View E是视图层级中包含触摸的最低的视图,因此它成为命中测试视图(hit-test view)。

    Hit-testing returns the subview that was touched

hitTest:withEvent:方法为给定的CGPoint和UIEvent返回一个点击测试视图(hit-test view)。hitTest:withEvent:方法首先调用pointInside:withEvent:方法。 如果传递到hitTest:withEvent:方法的点是在视图的边界内,pointInside:withEvent:返回YES。然后,在每个返回YES的子视图上递归调用hitTest:withEvent:方法 。

如果传递到hitTest:withEvent:方法的点不在视图的边界内,第一次调用pointInside:withEvent:方法返回 NO ,该点被忽略,hitTest:withEvent:返回nil 。 如果子视图返回NO,则视图层级结构的这个整个分支将被忽略,因为如果触摸没有发生在该子视图中,则它也不会出现在该子视图的任何子视图中。这意味着在子视图内而在父视图之外的任何点都不能接受点击事件,因为触摸点必须在父视图和子视图边界内。如果子视图的clipsToBounds属性设置为NO,则可能出现此问题。见示例将事件传递给子视图

注:触摸对象为其生命周期而关联到其命中测试视图(hit-test view),即使触摸稍后移动到视图之外。

命中测试视图(hit-test view)被给予首先处理触摸事件的机会。 如果命中测试视图(hit-test view)无法处理的事件,事件沿着响应者链向上传播(如响应者链由响应者对象组成中描述),直到系统找到一个可以处理它的对象。

2. 响应者链由响应者对象组成

许多类型的事件依赖于为事件传递的响应者链。 响应链是一系列被链接起来的响应对象。 它从第一响应者开始,到程序对象(UIApplication object)结束。 如果第一响应者不能处理事件,它转发事件到响应者链中的下一个响应者。

响应者对象是一个可以响应和处理事件的对象。 UIResponder类是所有响应者对象的基类,它不仅为事件处理定义编程接口,也为常见响应者行为定义编程接口。UIApplication, UIViewController和UIView类的实例都是响应者(responder),这意味着所有的视图和大多数控制器对象都是响应者。 注意核心动画层不是响应者。

第一个响应者被指定为第一个接收事件。 通常,第一响应者是视图对象。 一个对象通过做两件事情成为第一个响应者:

  1. 重写canBecomeFirstResponder方法返回YES。
  2. 接收becomeFirstResponder消息。 如果需要,对象可以向自身发送此消息。

注:请确保您的应用程序在指派一个对象成为第一个响应者之前已经建立了对象图(has established its object graph,个人感觉应该理解为对象已经被渲染完成)。 例如,您通常在重写的viewDidAppear:方法中调用becomeFirstResponder方法。 如果您尝试在viewWillAppear:中指派第一响应者,你的对象图尚未建立(object graph is not yet established,个人理解为对象渲染尚未完成),所以becomeFirstResponder方法返回 NO 。

事件不是唯一依赖响应者链的对象,响应者链用于以下所有情况:

  • 触摸事件(Touch events):如果命中测试视图(hit-test view)不能够处理触摸事件,事件以命中测试视图(hit-test view)为起点沿着响应者链向上传递。
  • 运动事件(Motion events):为了使用UIKit处理摇动动作事件,第一响应者必须实现UIResponder类的motionBegan:withEvent:motionEnded:withEvent:的方法。
  • 遥控事件(Remote control event):为了处理遥控事件,第一响应者必须实现UIResponder类的remoteControlReceivedWithEvent:方法。
  • 动作消息(Action messages):当用户操作一个控制对象,例如一个按钮(button)或者开关(switch),并且动作方法(action method)的目标(target)是nil,则消息以控制视图为起点沿着响应者链传递。参阅示例:将事件传递给父视图
  • 编辑菜单消息(Editing-menu messages):当用户点击编辑菜单中的命令,iOS使用响应者链找到实现了必要方法的对象(如cut:copy:paste: )。 想了解更多信息,请参阅显示和管理编辑菜单
  • 文本编辑(Text editing):当用户点击text field或text view,该视图自动成为第一个响应者。 默认情况下,虚拟键盘出现,text field或text view成为编辑的焦点。您可以显示自定义输入视图,而不是键盘。 您还可以向任何响应者对象添加自定义输入视图。 想了解更多信息,请参阅自定义数据输入视图

UIKit自动设置用户点击的text field或text view为第一个响应者; 应用程序必须使用becomeFirstResponder方法显式设置所有其他对象为第一响应者。

3. 响应者链遵循特定传递路径

如果初始对象(命中测试视图或第一个响应者)不处理事件,UIKit将事件传递给链中的下一个响应者。 每个响应者决定是否它要处理事件或通过调用其nextRsponder方法传递给它自己的下一个响应者。这种处理持续进行,直到一个响应者对象处理事件或有没有更多的响应者。

当iOS检测到事件并将其传递给初始对象(通常是视图)时,响应者链序列开始。 初始视图拥有第一机会处理事件。下图显示了两个不同配置应用程序的两个不同事件传递路径。应用程序的事件传递路径取决于其特定结构,但所有事件传递路径都遵循相同的探视程序。

The responder chain on iOS

对于左侧的应用程序,事件遵循以下路径:

  1. 初始视图试图处理该事件或消息。如果它不能处理这个事件,它将事件传递到其父视图 ,因为初始视图在它的视图控制器的视图层次中不是最顶部的视图。
  2. 父视图尝试处理该事件。如果父视图不能处理事件,它将事件传递到其超级视图,因为它仍然不是视图层次中最顶部的视图。
  3. 视图控制器的视图层次中最顶层视图尝试处理该事件。如果最顶层的视图不能处理事件,它将事件传递到它的视图控制器。
  4. 视图控制器尝试处理该事件,如果不能,将事件传递到窗口。
  5. 如果窗口对象不能处理该事件,传递事件到单例应用程序对象。
  6. 如果应用程序对象不能处理这个事件,它丢弃该事件。

右侧的应用程序遵循稍微不同的路径,但所有事件传递路径遵循以下探视程序:

  1. 视图在其视图控制器的视图层次结构上向上传递事件,直到它到达最顶层视图。

  2. 最顶层视图将事件传递到其视图控制器。

  3. 视图控制器将事件传递到其最顶层视图的父视图。

    重复步骤1-3,直到事件到达根视图控制器。

  4. 根视图控制器将事件传递到窗口对象。

  5. 窗口将事件传递给应用程序对象。

重要提示:如果您实现一个自定义视图来处理遥控事件,动作消息,UIKit的摇移动事件,或编辑菜单消息,不要直接转发事件或消息到nextResponder来沿响应者链向上传递。 相反,调用当前事件处理方法的超类实现,让UIKit处理响应者链的遍历。

三、应用

从事件传递与响应者链的内容思考一些应用例子。

1. 扩大视图的点击区域

一个按钮的尺寸是20*20,如果要扩大按钮的点击区域(上下左右各扩大10),有以下处理方法:

  • 按钮设置image,然后按钮的size设置的比实际大一倍。
  • 在按钮上覆盖一层较大的View或者Button,设置点击事件。
  • 自定义Button,覆盖hitTest:withEvent:或者pointInside:withEvent:方法。

我们只举例说明第三种方法:

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    NSLog(@"%s", __PRETTY_FUNCTION__);
    if (self.userInteractionEnabled == NO || self.hidden || self.alpha <= 0.01) {
        return nil;
    }
    
    CGRect responseRect = CGRectInset(self.bounds, -10, -10);
    if (CGRectContainsPoint(responseRect, point)) {
        for (UIView *subView in [self.subviews reverseObjectEnumerator]) {
            CGPoint convertedPoint = [subView convertPoint:point fromView:self];
            UIView *hitTestView = [subView hitTest:convertedPoint withEvent:event];
            if (hitTestView) {
                return hitTestView;
            }
        }
        return self;
    }
    return nil;
}

或者

- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event {
    NSLog(@"%s", __PRETTY_FUNCTION__);   
    CGRect bounds = CGRectInset(self.bounds, -10, -10);
    return CGRectContainsPoint(bounds, point);
}

2. 将事件传递给父视图

在controller中有一个YKNoteEventHandingView,其上面再添加一个YKNoteEventHandlingButton,点击Button将事件传递到View。有以下几种做法:

  • Button的- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event方法返回nil,hit-test view为父视图

  • YKNoteEventHandingView的- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event方法返回self,阻止事件传递给子视图

  • 设置Button的target为nil,Button无法处理事件响应,事件沿着响应者链向上传递,传递到父视图。示例如下

#import "YKNoteEventHandingView.h"

@implementation YKNoteEventHandingView
//在View中写一个action方法,判断View中的Button的target为nil的时候是否会执行,若执行,则消息沿着响应者链向上传递了
- (void)ykNoteEventHandlingGreenButtonDidTouchUpInside:(UIButton *)button {
    NSLog(@"%s \n %@", __PRETTY_FUNCTION__, button);
}

@end
  
#import "YKNoteEventHandlingButton.h"
//在Button中写一个action方法,判断Button的target为nil的时候是否会执行,若执行,则消息沿着响应者链传递了
@implementation YKNoteEventHandlingButton

- (void)ykNoteEventHandlingGreenButtonDidTouchUpInside:(UIButton *)button {
    NSLog(@"%s \n %@", __PRETTY_FUNCTION__, button);
}
#import "YKNoteEventHandingViewController.h"
#import "YKNoteEventHandingView.h"
#import "YKNoteEventHandlingButton.h"

@interface YKNoteEventHandingViewController ()

@property (nonatomic, strong) YKNoteEventHandingView *yKNoteEventHandingView;
@property (nonatomic, strong) YKNoteEventHandlingButton *ykNoteEventHandlingButton;

@end

@implementation YKNoteEventHandingViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    self.title = @"EventHandling";
    self.view.backgroundColor = [UIColor whiteColor];
    //View
    [self.yKNoteEventHandingView setFrame:CGRectMake(50, 100, 200, 200)];
    [self.view addSubview:self.yKNoteEventHandingView];

    //Button
    [self.ykNoteEventHandlingButton setFrame:CGRectMake(60, 60, 100, 100)];
    [self.yKNoteEventHandingView addSubview:self.ykNoteEventHandlingButton];
}

#pragma mark - event
- (void)ykNoteEventHandlingGreenButtonDidTouchUpInside:(UIButton *)button {
    NSLog(@"%s \n %@", __PRETTY_FUNCTION__, button);
}

#pragma mark - getter
- (YKNoteEventHandingView *)yKNoteEventHandingView {
    if (_yKNoteEventHandingView == nil) {
        _yKNoteEventHandingView = [[YKNoteEventHandingView alloc] init];
        _yKNoteEventHandingView.backgroundColor = [UIColor redColor];
    }
    return _yKNoteEventHandingView;
}

- (YKNoteEventHandlingButton *)ykNoteEventHandlingButton {
    if (_ykNoteEventHandlingButton == nil) {
        _ykNoteEventHandlingButton = [[YKNoteEventHandlingButton alloc] init];
        _ykNoteEventHandlingButton.backgroundColor = [UIColor greenColor];
        [_ykNoteEventHandlingButton addTarget:nil action:@selector(ykNoteEventHandlingGreenButtonDidTouchUpInside:) forControlEvents:UIControlEventTouchUpInside];
    }
    return _ykNoteEventHandlingButton;
}
  //Button的target设置为nil的时候,执行了YKNoteEventHandlingButton中的方法,说明target为nil的时候事件沿着响应者链传递了
  -[YKNoteEventHandlingButton ykNoteEventHandlingGreenButtonDidTouchUpInside:] 
   <YKNoteEventHandlingButton: 0x100224950; baseClass = UIButton; frame = (60 60; 100 100); opaque = NO; layer = <CALayer: 0x17002a1a0>>

  //注释掉Button中的方法。输出内容如下,说明事件沿着响应者链向上传递了。
  -[YKNoteEventHandingView ykNoteEventHandlingGreenButtonDidTouchUpInside:] 
   <YKNoteEventHandlingButton: 0x10030fe40; baseClass = UIButton; frame = (60 60; 100 100); opaque = NO; layer = <CALayer: 0x17003d520>>

  //注释掉Button和View中的方法。输出内容如下,说明事件沿着响应者链向上传递了。
  -[YKNoteEventHandingViewController ykNoteEventHandlingGreenButtonDidTouchUpInside:] 
   <YKNoteEventHandlingButton: 0x100402fd0; baseClass = UIButton; frame = (60 60; 100 100); opaque = NO; layer = <CALayer: 0x1740315a0>>

3. 将事件传递给兄弟视图

假设有下图所示的布局,我们希望点击view C的时候view B响应事件,而点击View D和View E的时候正常响应。这个时候通过重写view C的hittest可以解决这个问题,在C的hittest里面直接返回nil就行了。

Hit-testing returns the subview that was touched
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    NSLog(@"%s", __PRETTY_FUNCTION__);
    
    UIView *hitTestView = [super hitTest:point withEvent:event];
    if (hitTestView == self) {
        return nil;
    }
    return hitTestView;    
}

4. 将事件传递给子视图

如下图,banner为CollectionView中的一个楼层,CollectionViewCell中有个scrollView,scrollView中为图片,现在将cell的宽度缩小一半(变为蓝色框部分),设置cell和scrollview的clipsToBounds为NO,现在在右侧处滑动,scrollview中的图片显然不会滑动,因为不满足pointInside:withEvent:,这时只需要修改cell的- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event方法,返回scrollview即可。

传递事件到子视图
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    UIView *hitTestView = [super hitTest:point withEvent:event];
    if (hitTestView == nil) {
        hitTestView = self.scrollView;
    }
    return hitTestView;
}

参考:

https://developer.apple.com/library/content/documentation/EventHandling/Conceptual/EventHandlingiPhoneOS/Introduction/Introduction.html

https://developer.apple.com/library/content/documentation/EventHandling/Conceptual/EventHandlingiPhoneOS/event_delivery_responder_chain/event_delivery_responder_chain.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,298评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,701评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 107,078评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,687评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,018评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,410评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,729评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,412评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,124评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,379评论 2 242
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,903评论 1 257
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,268评论 2 251
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,894评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,014评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,770评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,435评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,312评论 2 260

推荐阅读更多精彩内容