事件传递响应链

什么是事件传递?

事件传递说白了就是iOS应用程序对用户操作进行逐级、有序处理的过程,这个过程会由UIWindow开始逐级向子视图进行检查。

第一步:捕捉用户操作

当用户触摸屏幕触发一个触摸事件(Touch Event)时,UIKit 会创建一个包含事件信息的 UIEvent 对象。然后把这个UIEvent对象放到到事件队列当中等待处理。对于触摸事件来说,这个 UIEvent对象是一系列触摸操作的集合。

单例的UIApplication对象会从事件队列(Event Queue)的顶部取出这个UIEvent对象。然后,这个对象会首先被送到当前应用的UIWindow对象,这个UIWindow对象会带着需要处理的UIEvent对象进行一个叫做hit-testing的过程。这个过程说白了就是寻找用户点击的位置到底是属于哪个View。

第二步:通过hit-testing寻找触摸目标

hitTest方法首先会掉用pointInside方法来判断触摸的位置是否在被检查的View中,如果是,那么pointInside方法返回True,然后继续使用hitTest方法检查该View的所有子视图。如果触摸位置不在被检查的View中,那么pointInside方法返回false,因而hitTest方法会返回nil。这样的话,不再检查该View的子类。这就意味着,如果要让某个View的事件被触发,必须保证该View在它的父视图当中

下面使用一个苹果文档中提供的例子来说明:

Hit-testing
Hit-testing

如果用户触摸View E,首先,检查View A,用户的操作处于View A,所以下一步检查View A的子类View B和View C。显然,触摸点不在View B中,所以开始检查View C中的两个View。最后,View E被判定为被用户点击的视图。

什么是响应链?

响应链是一系列连在一起的响应者对象,开始于第一响应者(First Respondor),结束于当前应用程序对象(UIApplication)。响应者对象就是能处理事件的对象,响应者对象的基类是UIResponder,所有响应者都是这个类的子类:

响应者类图

如果想要让某一View成为第一响应者,需要满足两个条件:重写继承自UIResponder的canBecomeFirstResponder()方法并返回true,并且接收一个来自becomeFirstResponder()方法的message。UIKit自动的将UITextfield和UITextView这两个控件设置为当用户点击时自动成为第一响应者,而其他的控件用户就需要自己设置becomeFirstResponder()了。

当我们经过上一步的hitTest方法检查出被点击的视图后,如果该视图不能处理该事件,那么就会使用nextResponder()方法将事件传递给响应链的下一个响应者对象,直到被处理或者没有下一个响应者。如果连UIApplication都处理不了事件,那么该事件将会被销毁。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一、简介 在我们点击屏幕的时候,iOS系统会获取“单击”行为,把这个信息包装成UITouch和UIEvent对象,...
    nuclear阅读 2,306评论 5 18
  • 序言 当我们在使用微信等工具,点击扫一扫,就能打开二维码扫描视图。在我们点击屏幕的时候,iphone OS获取到了...
    九洲仙人阅读 533评论 1 0
  • 序言 当我们在使用微信等工具,点击扫一扫,就能打开二维码扫描视图。在我们点击屏幕的时候,iphone OS获取到了...
    Johnny_Chang阅读 521评论 0 2
  • 当我们在使用微信等工具,点击扫一扫,就能打开二维码扫描视图。在我们点击屏幕的时候,iphone OS获取到了用户进...
    lbfly_boy阅读 678评论 0 0
  • 序言面试经常问道的一个问题:在一个app中间有一个button,在你手触摸屏幕点击后,到这个button收到点击事...
    Little_Shaun阅读 947评论 1 0