触摸事件

触摸事件的生命周期

  • 1.手指触摸屏幕产生一个事件

  • 2.系统响应

IOKit.framework将该事件封装为IOHIDEvent对象
IPC进程间通信,通过 mach port 转发给SpringBoad进程
触发主线程runloop的source1事件源回调
如果当前有app在前台运行则将触摸事件通过IPC传递给前台APP进程,否则由桌面系统处理

  • 3.app响应

APP进程的mach port接受到SpringBoard进程传递来的触摸事件,主线程runloop被唤醒,触发source1回调
source1回调又触发了一个source0回调,将接收到的IOHIDEvent对象封装成UIEvent对象
source0回调内部将触摸事件添加到UIApplication对象的事件队列中
事件离开队列后,UIApplication寻找一个最佳响应者(hit-testing)
结果:找到响应者被捕获消耗或者未找到响应者释放该响应
runloop进行休眠,等待下一个事件唤醒

mach port 进程端口,各进程之间通过它进行通信
SpringBoad 系统进程,可以理解为桌面系统,可以统一管理和分发系统接收到的触摸事件

为什么把事件放到队列当中,而不放到栈当中?

  • 队列是先进先出,栈是先进后出,先发生的事件当然首先处理,所以把事件放入到队列当中而不放到栈当中

寻找事件的最佳响应者

  • 事件的传递

    • UIApplication首先将事件传递给窗口对象(UIWindow),若存在多个窗口,则优先询问后显示的窗口
    • 若窗口不能响应事件,则将事件传递其他窗口;若窗口能响应事件,则从后往前询问窗口的子视图。
    • 子视图若不能响应,则将事件传递给上一个同级子视图;若能响应,则从后往前询问当前视图的子视图。
    • 视图若没有能响应的子视图了,则自身就是最合适的响应者

    自下而上,依次询问是否能够响应该事件
    优先询问后添加的子视图,即子视图数组中靠后的视图
    UIApplication ——> UIWindow ——> 子视图 ——> ...

视图如何判断能否响应事件

  • 每个UIView对象都有一个 hitTest:withEvent: 方法,这个方法是Hit-Testing过程中最核心的存在,其作用是询问事件在当前视图中的响应者,同时又是作为事件传递的桥梁。
hitTest:withEvent: 逻辑
若当前视图无法响应事件,则返回nil
若当前视图可以响应事件,但无子视图可以响应事件,则返回自身作为当前视图层次中的事件响应者
若当前视图可以响应事件,同时有子视图可以响应,则返回子视图层次中的事件响应者
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event{
    //3种状态无法响应事件
     if (self.userInteractionEnabled == NO || self.hidden == YES ||  self.alpha <= 0.01) return nil; 
    //触摸点若不在当前视图上则无法响应事件
    if ([self pointInside:point withEvent:event] == NO) return nil; 
    //从后往前遍历子视图数组 
    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) 
        {
            //如果子视图中有更合适的就返回
            return fitView; 
        }
    } 
    //没有在子视图中找到更合适的响应视图,那么自身就是最合适的
    return self;
}
  • pointInside:withEvent: 这个方法,用于判断触摸点是否在自身坐标范围内。默认实现是若在坐标范围内则返回YES,否则返回NO。

不能响应事件的条件

  • 不接收用户交互 (userInteractionEnabled = NO)
  • 隐藏 (hidden = YES)
  • 透明 (alpha = 0.0 ~ 0.01)

你可能感兴趣的:(触摸事件)