响应者链

UIResponder、UIEvent 和 UIControl

  • UIKit 中我们使用响应者对象(Responder)接收和处理事件。一个响应者对象一般是 UIResponder 类的实例,它常见的子类包括 UIViewUIViewControllerUIApplication,这意味着几乎所有我们日常使用的控件都是响应者,如 UIButtonUILabel 等等。
  • 在 UIResponder 及其子类中,我们是通过有关触摸(UITouch)的方法来处理和传递事件(UIEvent),具体的方法如下:
    1
    2
    3
    4
    - (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
    - (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
    - (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event
    - (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event

确定第一响应者

  • 实际上这个流程就是 UIView 的一个方法:hitTest:(CGPoint)point withEvent:(UIEvent *)event,方法最后返回的UIView即第一响应者,这个方法代码还原应该是这样的:
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    // 什么时候调用:只要事件一传递给一个控件,那么这个控件就会调用自己的这个方法
    // 作用:寻找并返回最合适的view
    // UIApplication -> [UIWindow hitTest:withEvent:]寻找最合适的view告诉系统
    // point:当前手指触摸的点
    // 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] == NO) return nil;
    // 3.从后往前遍历子控件数组
    int count = (int)self.subviews.count;
    for (int i = count - 1; i >= 0; i--) {
    // 获取子控件
    UIView *childView = self.subviews[i];
    // 坐标系的转换,把窗口上的点转换为子控件上的点
    // 把自己控件上的点转换成子控件上的点
    CGPoint childP = [self convertPoint:point toView:childView];
    UIView *fitView = [childView hitTest:childP withEvent:event];
    if (fitView) {
    // 如果能找到最合适的view
    return fitView;
    }
    }
    // 4.没有找到更合适的view,也就是没有比自己更合适的view
    return self;
    }

事件传递

  1. 首先看initial view能否处理这个事件,如果不能则会将事件传递给其上级视图(inital view的superView);
  2. 如果上级视图仍然无法处理则会继续往上传递;一直传递到视图控制器view controller,首先判断视图控制器的根视图view是否能处理此事件;如果不能则接着判断该视图控制器能否处理此事件,如果还是不能则继续向上传递;(对于第二个图视图控制器本身还在另一个视图控制器中,则继续交给父视图控制器的根视图,如果根视图不能处理则交给父视图控制器处理);
  3. 一直到 window,如果window还是不能处理此事件则继续交给application处理,如果最后application还是不能处理此事件则将其丢弃

参考文章

史上最详细的iOS之事件的传递和响应机制-原理篇