了解IOS事件处理的本质关键要先掌握几个概念。首先是事件的派发(Event Delivery)的过程, 一个是响应者链条如何构成。
S1: 正是因为当我们点击屏幕上某个点的时候, IOS会检查到手指触摸操作(Touch),并生产一个UITouch对象,将其打包成一个UIEvent对象。然后将其放入当前活动的Application的事件对列, UIApplication会从事件对列中按照对列的顺序,取出触摸事件传递给UIWindow处理,UIWindow对象会使用hitTest:withEvent:方法来寻找此次的触摸操作初始点所在的最深层次的视图(View). **即调用hitTest:withEvent会返回该触摸点所在的最深层次的视图。 **
S2: 这就要说到深度优先搜索算法,hitTest:withEvent正是基于深度优先搜索的方式来找到最深层次的视图对象。所以我来介绍以下深度优先算法的思想, 要理解该思想, 你首先要有树结构这一概念(参见数据结构中的树结构)。该思想是从根节点开始遍历树,而遍历的顺序是采用把下一个子节点当做当前根节点继续遍历。 所以其是先遍历到最树的最深的一层再层层回朔到根节点,接着在把另外一个子节点当做当前根节点继续遍历。正是基于这种思想, 所以我们可以很方便的采用递归来实现。 如果还是不理解,有两种办法帮助你,一种是去找深度优先的动态图,一看就懂了我说的。另外一种方法是去复习数据结构与算法。
S3:根据官方文档给出的条件是(hidden == YES || userInteractionEnabled == YES || alpha < 0.01 || subViews.bounds > subViews.superView.bounds)
S4: 首先先明确何为响应者? ===> 在ios开发中继承自UIResponder的类或子类就是响应者,顾明思意,响应者是用来相应事件的(触摸事件、运动事件、远程遥控事件)。所以所谓的响应者链条就是一系列响应者构成的层次结构。
S5: 响应者链条是通过nextResponder方法的返回值来组成这种层次结构的 ,苹果有一段官方解释如下:
The UIResponder class does not store or set the next responder automatically, instead returning nil by default. Subclasses must override this method to set the next responder. UIView implements this method by returning the UIViewController object that manages it (if it has one) or its superview (if it doesn’t); UIViewController implements the method by returning its view’s superview; UIWindow returns the application object, and UIApplication returns nil.
也就是说,响应者对象是不会自动设置和存储下一个响应者,默认情况下是直接返回nil。而继承自UIResponder的子类必须重写这个方法来设置下一个响应者,并且需要遵循如下规范
1. 如果子类是UIView,那么其getter方法的nextResponder必须返回其UIViewController对象。
如果不存在控制器,则返回其父视图对象。
2. 如果子类是UIViewController对象, 那么重写的nextResponder方法必须返回其view视图的父视图对象。
3. 如果子类是UIWindow对象,那么重写的nextResponder方法返回的是application对象
4. 如果子类是UIApplication对象,那么重写的nextResponder方法,返回nil。
通过上述规范,结合下图,你应该能很容易理解所谓的响应者链条如何构成:
#import "SWPButton.h"
@implementation SWPButton
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
[self.nextResponder touchesBegan:touches withEvent:event];
}
@end
所以此时如果你点击按钮, 其会先调用这个自定义按钮的touchesBegan, 因为点击事件传递到了button身上,所以会调用touchesBegan来响应该事件,但是该事件不在交给父类处理,所以不会调用action。只会继续将其抛给上一个响应者。
@implementation UIView (ParentController)
-(UIViewController*)parentController{
UIResponder *responder = [self nextResponder];
while (responder) {
if ([responder isKindOfClass:[UIViewController class]]) {
return (UIViewController*)responder;
}
responder = [responder nextResponder];
}
return nil;
}
@end
思路很简单,就是利用响应者链条来寻找UIViewController.
那么这个UIView的上一个响应者只有两种情况,一种是依然是一个UIView对象或其子对象,也就是说它是这个UIView的子View。
一种是这个 UIView的控制器。所以我们才需要循环判断,然后不断找(循环),直到第一次找到的控制器,就是这个UIView所在的控制器。
如果找不到,就返回nil。
是不是更加理解你事件派发的过程,所谓的事件派发过程,其实就是寻找最合适的视图的时候,事件随着这个寻找过程,不断传递。 为什么要传递UIEvent呢?因为通过它给以获得Touch对象,而通过Touch对象我们可以获得初始触摸点。也就是说hitTest:withEvent主要实现的功能是,传递事件,找到最合适的视图。