事件响应链

一、相关类介绍

1、UIResponder

      iOS中只有继承了 UIResponder 的对象才能接收并处理事件。UIResponder 的派生体系如下:

下面是 UIResponder 的接口:

 @interface UIResponder : NSObject

// 响应者相关的属性和方法

@property(nonatomic, readonly, nullable) UIResponder *nextResponder;

@property(nonatomic, readonly) BOOL canBecomeFirstResponder; 

- (BOOL)becomeFirstResponder; @property(nonatomic, readonly) BOOL canResignFirstResponder; 

- (BOOL)resignFirstResponder; 

@property(nonatomic, readonly) BOOL isFirstResponder; 

 // 触摸事件 

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

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

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

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

- (void)touchesEstimatedPropertiesUpdated:(NSSet*)touches NS_AVAILABLE_IOS(9_1);

// 关于3Dtouch的按压事件 

- (void)pressesBegan:(NSSet*)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0); 

- (void)pressesChanged:(NSSet*)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0); 

- (void)pressesEnded:(NSSet*)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0); 

- (void)pressesCancelled:(NSSet *)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0);

// 加速计事件

- (void)motionBegan:(UIEventSubtype)motion withEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(3_0);

- (void)motionEnded:(UIEventSubtype)motion withEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(3_0);

- (void)motionCancelled:(UIEventSubtype)motion withEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(3_0);

// 远程事件

- (void)remoteControlReceivedWithEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(4_0);

- (BOOL)canPerformAction:(SEL)action withSender:(nullable id)sender NS_AVAILABLE_IOS(3_0);

// 其它

- (nullable id)targetForAction:(SEL)action withSender:(nullable id)sender NS_AVAILABLE_IOS(7_0);

@property(nullable, nonatomic,readonly) NSUndoManager *undoManager NS_AVAILABLE_IOS(3_0);

@end

1.1、UIResponder 遵守的协议 UIResponderStandardEditActions,负责处理 复制、粘贴 等操作。 

1.2、UIResponder 可以处理4种事件,分别是:触摸事件、按压事件(3Dtouch)、加速计事件(摇一摇)、远程事件(耳机操控事件),当事件发生时,UIResponder 就会调用对应的方法。 

1.3、处理事件时涉及2个对象:UITouch、UIEvent。(UIPress、UIPressesEvent不作介绍)  

1.3.1、UITouch 

UITouch是处理手指触屏交互的底层对象,保存着对应的时间、阶段等相关信息。当有一根手指触摸屏幕时就会产生一个UITouch对象,当该手指移动时,该UITouch对象的属性就会不断更新,当手指离开屏幕时,该UITouch对象就会消亡,所以其生命周期是从手指接触屏幕开始,到手指离开屏幕结束。下面是相关属性和方法: 

// 触摸点所在的窗口 

@property(nonatomic,readonly,retain) UIWindow *window; 

// 触摸点所在的视图 

@property(nonatomic,readonly,retain) UIView *view; 

// 短时间内点按屏幕的次数,可以根据tapCount判断单击、双击或多点击 

@property(nonatomic,readonly) NSUInteger tapCount; 

// 触摸事件产生或变化时的时间,单位是s 

@property(nonatomic,readonly) NSTimeInterval timestamp; 

// 当前触摸事件的状态 

@property(nonatomic,readonly) UITouchPhase phase; 

// 触摸点在view上的位置(左上角为原点),参数view为nil时,返回触摸点在UIWindow上的位置 

- (CGPoint)locationInView:(UIView *)view; 

// 前一个触摸点的位置 

- (CGPoint)previousLocationInView:(UIView *)view; 

1.3.2、UIEvent 

UIEvent表示事件对象,记录事件产生的时刻和类型,可以将短时间内发生的触摸屏幕的多个UITouch视为一个事件,看作一个特定的手势或动作。UIEvent对象的生命周期也是从手指触摸屏幕开始到手指离开屏幕结束(如果有多个手指,以这些手指触屏引发事件开始,到任何一个手指离开屏幕导致事件结束为止)。下面是相关的属性和方法: 

// 事件类型 

@property(nonatomic,readonly) UIEventType type NS_AVAILABLE_IOS(3_0); 

@property(nonatomic,readonly) UIEventSubtype subtype NS_AVAILABLE_IOS(3_0); 

// 事件产生的时间 

@property(nonatomic,readonly) NSTimeInterval timestamp; 

// 触摸对象的集合 

@property(nonatomic, readonly, nullable) NSSet*allTouches; 

// 获取对应窗口上的触摸对象的集合 

- (nullable NSSet*)touchesForWindow:(UIWindow *)window; 

// 获取对应视图上的触摸对象的集合 

- (nullable NSSet*)touchesForView:(UIView *)view; 

// 获取对应手势识别器中的触摸对象的集合 

- (nullable NSSet*)touchesForGestureRecognizer:(UIGestureRecognizer *)gesture NS_AVAILABLE_IOS(3_2); 

// - (nullable NSArray*)coalescedTouchesForTouch:(UITouch *)touch NS_AVAILABLE_IOS(9_0); 

- (nullable NSArray *)predictedTouchesForTouch:(UITouch *)touch NS_AVAILABLE_IOS(9_0);

2、UIGestureRecognizer

      UIGestureRecognizer 是专门针对 UIView 的手势识别器,用于识别并处理特定手势,一个手势包含一个或多个 UITouch。其派生体系如下:

二、事件传递过程

      iOS是依靠“事件响应链”进行事件传递的,所谓“事件响应链”是一个由许多 UIResponder 对象组成的链条,要搞清楚这个链条,先来看看APP的视图结构,如下图:

      APP的生成次序是:首先生成 AppDelegate & UIApplication,然后添加 UIWindow,再添加 rootViewController(UIViewController),最后 rootViewController 的 view 添加各个子 view,这样就形成一个视图层级关系,事件响应链就依赖这种关系,那么一条响应链的结构顺序就是:

AppDelegate -> UIApplication -> UIWindow -> UIViewController -> 根view -> 子view1 -> 孙子view1 ->>>

在这条响应链中,前一个对象依次是后一个对象的下一个响应者(即nextresponder),它们之间的这种关系是在 setRootViewController && addsubview &&  viewcontroller初始化的时候形成的。

比如执行方法: [viewA addsubview:viewB];  后,

就会有:viewB.nextResponder = viewA。

      可见每一个视图的父视图是唯一的,每一个视图的下一个响应者也是唯一的,但是每一个父视图就可以有好多子视图,可以有好多上一个响应者,因此一个APP可以有多条事件响应链,最终形成一个树形结构。

      下面来看一条事件响应链的响应过程,以上图为例,当一个事件(UIEvent)发生时,系统首先会将其传递给最上层的响应者(通常是处于视图层级最顶部对应位置的视图),这个响应者就是一条响应链中的一个节点,如果该响应者能够处理事件,则处理该事件并终止事件传递,如果不能处理,事件就会沿着这条响应链向上(向AppDelegate那个方向)传递,直到遇到能够处理该事件的响应者,最终如果事件未能被处理就会被丢弃。下面是具体步骤:

(1) 事件(UIEvent)发生。

(2) initial view 尝试处理,如果不能处理,则传递给其父视图(superview)。

(3) 父视图尝试处理,如果不能处理,则继续传递给其父视图。

(4) 父视图尝试处理,如果不能处理,则传递给它所在的视图控制器。

(5) 视图控制器尝试处理,如果仍不能处理,则将事件传递给 UIWindow。

(6) UIWindow 尝试处理,如果不能处理,则传递给 UIApplication。

(7) UIApplication 尝试处理,如果不能处理,则传递给 AppDelegate。

(8) AppDelegate 尝试处理,如果也不能处理,则丢弃该事件。

需要注意一点:当 UIView 添加了手势识别器,事件传递到该 view 的时候,那么该 view 的手势识别器会优先处理该事件,比如:UIButton 添加点击事件和tap手势,tap手势优先处理事件(另外,如果 UIButton 重写 touchesBegan:withEvent: 等方法时,addTarget:action:forControlEvents: 方法无效),并且如果上层视图没有添加手势识别器的话,下层视图的手势事件任然会执行。

      上面说过,一个APP可能有好多条事件响应链,那么事件发生时,执行哪一条呢?这就要归功于它的原理事件分发机制(hit-Testing)。当手指触摸屏幕时,UIKit 就会产生一个事件(UIEvent)对象,并把它添加到事件队列中,然后由 UIApplication 从事件队列中取出首先分发给 UIWindow,从 UIWindow 开始进行分发,分发是依靠 UIView 的两个分类方法 hitTest:withEvent: & pointInside:withEvent: 以及 UIView 的 userInteractionEnabled、hidden、alpha 属性来进行的,以下图为例:

假如触摸事件发生在 ViewE 区域内,事件分发机制将执行如下步骤:

(1) 从 window 开始一直传递到 ViewA,此过程和下面过程一样,不作详细说明。

(2) ViewA 调用 hitTest:withEvent: 方法,该方法内部做了一些判断,先检查 ViewA 的 userInteractionEnabled=YES、hidden=NO、alpha>0.01 三个条件是否满足要求,然后调用 pointInside:withEvent: 方法判断触摸点是否在该 UIView 的区域内,当所有检查合格后,事件分发给它的子视图 ViewB & ViewC,也就是 ViewB & ViewC 调用 hitTest:withEvent: 方法。

(3) 假如 ViewB 在 ViewC 的顶部,那么 ViewB 先执步骤(2),不满足要求,那么 ViewC 继续执步骤(2),满足,然后 ViewC 将事件分发给其子视图 ViewD & ViewE,如果 ViewC 在 ViewB 的顶部,那么 ViewC 先执行步骤(2),因满足要求,固 ViewB 就会被忽略,不再执行步骤(2)。

(4) ViewD 和 ViewE 的执行步骤同步骤(3),最后因为 ViewE 没有子视图,所以就成为第一响应者了。

点击这里获取验证代码

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

推荐阅读更多精彩内容