1. 响应者链
用户触摸屏幕后,手机硬件受到触摸事件,分发给UIApplication,之后逐步传递到UIWindow->UIView->subViews(该数组从后向前遍历,即用户看到屏幕最上层的视图,这就解释了为什么用户点击事件能及时响应了)。
// point是该视图的坐标系上的点
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event
{
// 1.判断自己能否接收触摸事件
if (self.userInteractionEnabled == NO || self.hidden == YES || self.alpha <= 0.01) return nil;
// 2.判断触摸点在不在自己范围内
if (![self pointInside:point withEvent:event]) return nil;
// 3.从后往前遍历自己的子控件,看是否有子控件更适合响应此事件
int count = self.subviews.count;
for (int i = count - 1; i >= 0; i--) {
UIView *childView = self.subviews[i];
CGPoint childPoint = [self convertPoint:point toView:childView];
UIView *fitView = [childView hitTest:childPoint withEvent:event];
if (fitView) {
return fitView;
}
}
// 没有找到比自己更合适的view
return self;
}
hitTest方法会首先判断触摸点是否在self视图(当前视图),如果不在,直接返回nil,则self及其所有的子视图都被忽略处理该事件,此为第一层判断(1);如果当前点在self视图,则遍历subviews数组,如果数组中有view能够响应该触摸事件,则返回该子view(fitView),此为第二层判断(2);如果self的所有子视图都不能响应触摸事件,则返回self,由self视图处理触摸事件,此为第三层判断(3)。
第一层于第三层都好理解,但是第二层使用了递归去寻找最合适的子View,有点绕。着重分析下第二层,遍历subviews数组时,遍历的每个元素(从后向前,最后加入的view)执行hitTest时,又会从第一层开始,如果返回nil,那么就从该元素的兄弟视图开始执行hitTest,如果找到了合适的兄弟视图能够响应hitTest,则直接返回,所有的递归操作结束,返回值会一直回传到最开始分发事件的hitTest;如果没有找到则转入第三层,返回self。