iOS_响应者链 Responder Chain

文章目录

简述

  • 传递链:

    系统向离用户最近的view传递。

    UIKit --> active app's event queue --> window --> root view --> ... --> lowest view

  • 响应链:

    由离用户最近的view向系统传递。

    initial view --> super view --> ... --> view controller --> window --> Application --> AppDelegate

iOS中,只有继成UIResponder的对象才能接收并处理事件,UIReaponder是所有响应对象的基类,在UIResponder类中定义了处理上述各种事件的接口。UIApplication、UIViewController、UIWindow和所有继承直UIView和UIKit类都直接或间接的继承UIResponder,所以它们的实例都是可以构成响应者链的响应者对象。


Hit-Test 机制 (找到最佳响应者)

用户触摸,Touch -> UIEvent -> eventQueue -> UIApp -> UIWindow -> hitTest:withEvent: 查找触摸点所在视图。

当用户触摸屏幕时,系统首先要找到响应者Responder。将Touch以UIEvent的方式加入到UIApplication的事件队列中。UIApplication从事件队列中取出最新的事件传递到UIWindow。UIWindow通过 hitTest:withEvent: 方法寻找触摸点所在的视图,这个过程称之为hit-test view。

UIApplication -> UIWindow -> RootView -> supView -> View

在顶级视图(Root View)上调用pointInside:withEvent:方法判断触摸点是否在当前视图内;

如果返回NO,那么hitTest:withEvent:返回nil;

如果返回YES,那么它会向当前视图的所有子视图发送hitTest:withEvent:消息,所有子视图的遍历顺序是从最顶层视图一直到到最底层视图,即从subviews数组的末尾向前遍历,直到有子视图返回非空对象或者全部子视图遍历完毕。

响应者链 Responder chain

当系统通过Hit-Test机制找到触摸到的View,但是view没有或无法正确处理此次Touch。此时,系统便会通过响应者链寻找下一个响应者,对此次Touch进行相应:

View -> View Controller(如果存在)-> superView -> RootView -> UIWindow -> UIApplication


hitTest 实现原理大致如下:

objectivec 复制代码
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
  // 1、控件不允许与用户交互
  if (self.userInteractionEnabled == NO ||
      self.alpha <= 0.01 ||
      self.hidden == YES) {
    return nil;
  }
  // 2、点击的point不在当前控件内
  if (![self pointInside:point withEvent:event]) {
    return nil;
  }
  // 3、倒序遍历每一个子控件
  for (int i = (int)self.subviews.count - 1; i >= 0; i--) {
    UIView *childView = self.subviews[i];
    // 当前触控Point的坐标转换为相对于子控件的坐标
    CGPoint childPoint = [self convertPoint:point toView:childView];
    // 在子控件中找能响应的子控件(递归循环),从上层找起
    UIView *fitView = [childView hitTest:childPoint withEvent:event];
    if (fitView) {
      return fitView;
    }
  }
  // 4、子视图中没有能响应的view,就返回自己
  return self;
}

// 该方法判断触摸点是否在控件上,point必须是方法调用者的坐标
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event {
    return NO;
}

应用

获取当前View的控制器对象:

objectivec 复制代码
- (UIViewController *)getCurrentViewController{ 
    UIResponder *next = [self nextResponder]; 
    do { 
        if ([next isKindOfClass:[UIViewController class]]) { 
            return (UIViewController *)next; 
        } 
        next = [next nextResponder]; 
    } while (next != nil); 
    return nil; 
}

手势穿透:

objectivec 复制代码
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    UIView *view = [super hitTest:point withEvent:event];
    if (self == view) {
        return nil;
    }
    return v;
}
相关推荐
仙剑魔尊重楼4 小时前
iMazing 3.1.3官方中文版新功能介绍
macos·objective-c·cocoa
unable code7 小时前
磁盘取证-Flying_High
网络安全·ctf·misc·1024程序员节·磁盘取证
2501_915918418 小时前
HTTPS 代理失效,启用双向认证(mTLS)的 iOS 应用网络怎么抓包调试
android·网络·ios·小程序·https·uni-app·iphone
Swift社区8 小时前
Flutter 路由系统,对比 RN / Web / iOS 有什么本质不同?
前端·flutter·ios
zhyongrui8 小时前
SnipTrip 发热优化实战:从 60Hz 到 30Hz 的性能之旅
ios·swiftui·swift
Andy Dennis8 小时前
ios开发 xcode配置
ios·cocoa·xcode
JoyCong19989 小时前
iOS 27 六大功能前瞻:折叠屏、AI Siri与“雪豹式”流畅体验,搭配ToDesk开启跨设备新协作
人工智能·ios·cocoa
linweidong9 小时前
屏幕尺寸的万花筒:如何在 iOS 碎片化生态中以不变应万变?
macos·ios·移动开发·objective-c·cocoa·ios面试·ios面经
Cestb0n9 小时前
iOS 逆向分析:东方财富请求头 em-clt-auth 与 qgqp-b-id 算法还原
python·算法·ios·金融·逆向安全
00后程序员张11 小时前
无需越狱,来对 iOS 设备进行调试、管理与分析
android·ios·小程序·https·uni-app·iphone·webview