ios 响应链

首先清楚两个概念
响应者:对用户交互动作事件进行响应的对象。
响应者链:成为处理事件的响应者的先后顺序链。
平时当我们点击屏幕的时候,操作系统把包含点击事件的信息包装成UITouch和UIEvent形式的实例,然后找到当前运行的程序,逐级寻找能够响应这个事件的对象,直到没有响应者响应。这个寻找过程,就叫做事件的响应链,如下图所示


1452653940101147.png

当一个常见的用户事件发生的时候,UIKit会创建一个事件对象Event Object,该对象包含了事件处理所必须的一些信息。然后它会将事件对象置于激活的app事件队列。
一、Hit-Test机制
当用户触摸屏幕进行交互时,系统首先要找到响应者(Responder)。系统检测到手指触摸(Touch)操作时,将Touch以UIEvent的方式加入UIApplication事件队列中。UIApplication从事件队列中取出最新的触摸事件进行分发传递到UIWindow进行处理。UIWindow会通过hitTestL:withEvent:方法寻找触碰点所在的视图,这个过程被称之为hit-test-view(命中测试view)。
使用hit-testing去寻找触摸发生下的view。命中测试会执行检测判断是否该触摸点发生在某个具体的view的相对边界之内。如果检测是的,它就会递归传递的去检测该view的所有子view。该view的层级最底端view包含触摸点,它就成为“命中测试view”。之后iOS就会递交触摸事件给它处理。
举例说明:如下图

652024-27b51a71f8a42474.png

1.检测触摸点是否在view A的边界之内,如果是它会检测子视图B和C。
2.触摸点不在view B的边界之内,但是它在view C的边界之内,因此它就去检测C的子视图D和E。
3.触摸事件不在view D的边界之内,但是在view E的边界之内。view E是整个包含触摸事件的view层级中最底端的view,因此view E就成为hit-test-view。

其检测方法是:方法hitTest:withEvent传入CGPoint和UIEvent返回“命中测试view”。这个方法hitTest:withEvent开始执行通过调用自身的pointInside:withEvent方法。如果传入hitTest:withEvent的point在view的边界之内。pointInside:withEvent方法就会返回YES。然后该方法会递归调用hitTest:withEvent方法在每一个返回YES的子view上。

如果传入的hitTest:withEvent的点不在view的边界之内,首先会调用ponitInside:withEvent方法返回NO,该point会被忽略,并且hitTest:withEvent返回nil(注意这个过程,子视图也是根据pointInside:withEvent:的返回值来确定是返回空还是当前子视图对象的。并且这个过程中如果子视图的hidden=YES、userInteractionEnabled=NO或者alpha小于0.1都会并忽略),如果一个子view返回NO,那么整个view层级的分支都会被忽略,因为如果触摸点没有发生在子view,她也不可能发生在任何子view的子view。这就意味着任何子view上面的点都在边界之外并且它的父view也不能接受触摸事件,因为触摸点必须同时在父view和子view的边界之内。这种情况可能会发生,如果子view的 clipsToBound 属性设为NO。
二 、响应者链
有些时候,Touch后系统通过hit test 机制找到了触碰到的Initial View,但是Initial view并没有或者无法正常处理此次Touch。这个时候,系统便会通过响应者链寻找下一个响应者,以对此次Touc 进行响应。

Initial View -> View Controller(如果存在) -> superview -> · ·· -> rootView -> UIWindow -> UIApplication


1467190622568169.jpg

如果一个View有一个视图控制器(View Controller),它的下一个响应者是这个试图控制器,紧接着才是它的父视图(super view),如果一直到Root View都没有处理这个事件,事件会传递到UIWindow(iOS中有个单例Window),此时Window如果也没有处理事件,便进入UIApplication,UIApplication是一个响应者链的终点,它的下一个响应者指向nil,以结束整个循环。

在某些情景下,我们在点击子视图的时候仍然需要调用父视图的touchesBegan:withEvent:等方法,例如我们在父视图上添加了一个覆盖范围了父视图大部分面积的TableView或ScrollerView 或其他View,而我们要通过父视图的touchesBegan:withEvent:方法来收键盘。
这个时候我们可以通过UIView的类别,重写touch相关方法,代码如下:
-(void)setEnableNextResponder:(BOOL)enableNextResponder {
objc_setAssociatedObject(self, &enableNextResponderKey, enableNextResponderKey, OBJC_ASSOCIATION_ASSIGN);
}
-(BOOL)enableNextResponder {
return objc_getAssociatedObject(self, &enableNextResponderKey);
}
-(void)touchesBegan:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event{
if (self.enableNextResponder) {
[[self nextResponder] touchesBegan:touches withEvent:event];
[super touchesBegan:touches withEvent:event];
}
}
-(void)touchesEnded:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event {
if (self.enableNextResponder) {
[[self nextResponder] touchesEnded:touches withEvent:event];
[super touchesEnded:touches withEvent:event];
}
}
-(void)touchesMoved:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event {
if (self.enableNextResponder) {
[[self nextResponder] touchesMoved:touches withEvent:event];
[super touchesMoved:touches withEvent:event];
}
}
-(void)touchesCancelled:(NSSet<uitouch *> *)touches withEvent:(UIEvent *)event {
if (self.enableNextResponder) {
[[self nextResponder] touchesCancelled:touches withEvent:event];
[super touchesCancelled:touches withEvent:event];
}
}</uitouch *></uitouch *></uitouch *></uitouch *>
三、响应链应用
重写PointInside
可以通过重写查找事件处理者的方法来实现不规则形状点击。最常见的不规则视图就是圆形视图,在demo中我设置view的宽高为200,那么重写方法事件如下:

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

推荐阅读更多精彩内容