iOS中触摸事件的传递和响应分析

当ios一次触摸事件发生时,它是如何传递给视图,视图又对此次触摸事件如何响应?

首先我们需要了解应用程序接收到触摸事件后传递过程,概述如下:

当用户触摸屏幕后产生的一次触摸事件会被系统加入到当前活动的UIApplication管理的事件队列中,UIApplication会从事件队列按续取出事件,将事件一层一层地传递给应用程序的主窗口及视图层级中的事件响应者。

事件的产生源于触摸:UITouch

用户用一根手指触摸屏幕,会产生一个UITouch对象,这个对象记录了触摸的相关信息,包括:触摸产生的时间、触摸状态、点击次数、触摸所在的窗口及视图等。我们可以通过UITouch的属性获得:

@interface UITouch : NSObject
@property(nonatomic,readonly) NSTimeInterval      timestamp;
@property(nonatomic,readonly) UITouchPhase        phase;
@property(nonatomic,readonly) NSUInteger          tapCount;
@property(nullable,nonatomic,readonly,strong) UIWindow *window;
@property(nullable,nonatomic,readonly,strong)  UIView *view;
@end

01事件本身:UIEvent

系统将用户的一次触摸产生的事件打包成UIEvent对象,它记录了事件的类型、事件产生的时间、以及所有的UITouch对象等。

@interface UIEvent : NSObject
@property(nonatomic,readonly) UIEventType     type API_AVAILABLE(ios(3.0));
@property(nonatomic,readonly) NSTimeInterval  timestamp;
@property(nonatomic, readonly, nullable) NSSet <UITouch *> *allTouches;

应用程序可以接收不同类型的事件,触摸事件是最常见的事件,它会传递给最初发生触摸的视图。触摸事件UIEvent对象可以包含一个或多个触摸,并且每个触摸都由一个UITouch对象表示。

在这里我们讲述的触摸事件,由UIEventType的api中UIEventTypeTouches可以体现。Api中还体现了iOS中的事件类型如移动事件、远程控制事件、滚动事件等。

typedef NS_ENUM(NSInteger, UIEventType) {
    UIEventTypeTouches,
    UIEventTypeMotion,
    UIEventTypeRemoteControl,
    UIEventTypePresses,
    UIEventTypeScroll,
    UIEventTypeHover,
    UIEventTypeTransform,
};

02事件响应者:UIResponder

能够接收事件、处理事件的对象都是响应者对象,都是UIResponder的子类对象。其中UIView、UIViewController、UIApplication的实例对象都是响应者对象。以下是UIResponder用于处理触摸事件的方法:

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;//触摸开始
- (void)touchesMoved:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;//移动
- (void)touchesEnded:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;//结束
- (void)touchesCancelled:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;//取消

一次触摸事件产生,系统自动调用以上处理事件的方法,touches参数中存储着UITouch触摸对象集合,UITouch对象记录了触摸的相关信息;event参数存储的UIEvent事件对象即为一次触摸事件。

事件的传递

UIApplication将事件传递应用程序的主窗口,主窗口会继续按视图层级在子视图中向上寻找某个视图来处理触摸事件,而子视图众多,谁才是最终处理事件的视图,这个事件逐级传递的过程形成了事件的传递链。

事件传递的目的是为了找到事件的最佳响应者,过程如下,我们以UIView视图为例:

应用程序将事件传递给主窗口,主窗口判断自身能否响应事件、判断触摸点是否在自身范围内,如果能响应且在自身范围内,则在它的子控件数组中从后添加的子视图向先添加的子视图去询问,这些子视图能否响应事件及触摸点是否在其范围内,即如果某个子视图无法响应或不在其范围内,则去询问它的上一个被添加的同级子视图,如果当前视图能响应且在自身范围内,则继续向视图层级的下一级去做相同的判断,直到遍历到最后,视图没有没有符合上述条件的子视图时,那么它本身就是最佳响应事件的视图。

而寻找哪个视图来响应事件的核心方法是UIView对象的方法:

- (nullable UIView *)hitTest:(CGPoint)point withEvent:(nullable UIEvent *)event

pointInside则是判断事件是否发生在自身范围内:

- (BOOL)pointInside:(CGPoint)point withEvent:(nullable UIEvent *)event;

事件被传递给一个视图,则视图的hitTest方法被调用,它返回的是一个UIView对象,它会调用pointInside:withEvent:来判断触摸点是否在自身范围内, 如果pointInside:withEvent:返回YES,则从后往前遍历当前视图的子视图数组,目的是为了找到能够响应事件的视图,并且把事件传递下去,直到找到触摸点所在的子视图,则将子视图返回;如果在对当前视图的子视图数组的遍历中,没有找到触摸点所在的子视图,那么触摸点就在当前视图自身范围内,则返回当前视图。

但如果pointInside:withEvent:返回NO,即触摸点不在自身范围,或当前视图处于不可交互状态、隐藏状态或透明状态时,hitTest方法则返回nil,具体状态值如下:

  • 不允许交互:view.userInteractionEnabled = NO;
  • 隐藏:view.hidden = YES ;
  • 透明度:alpha 小于 0.01 时。

图例分析:

视图被添加到主窗口上的顺序即为由1到5,用户在第4个视图上红点位置进行了点击:

1.firstView对象的hitTest方法被调用,红点在firstView范围内,pointInside:withEvent:返回YES,会继续向firstView的子视图数组secondView和thirdView中查找。

2.因为thirdView是后添加到firstView上的,所以会首先在thirdView上调用hitTest方法,来寻找触摸点是否在其范围内,很显然pointInside:withEvent:返回YES,继续向它的子视图fifthView调用hitTest方法寻找触摸点没有在fifthView范围内。

3.继续遍历thirdView的其他子视图fourthView,pointInside:withEvent:返回YES。fourthView没有子视图,至此hitTest方法就会返回fourthView,作为事件传递链终端处理事件的视图即最佳响应者。

事件的响应

hitTest返回了处理事件的UIView对象,它是UIResponder的子类对象,是事件的响应者,能够接收和处理事件,他可以调用touchesBegan/touchesMove/touchesEnded/ touchesCancelled方法去处理事件,也可以把事件传递给其他响应者,touches方法默认把事件沿着响应者链条向父视图传递,自此事件在众多响应者之间的传递过程引申出一个概念“响应者链”,即由众多响应者构成的事件传递的链条。

事件在响应者链中的传递是为了使响应者对事件做出响应,UIResponder提供一个属性nextResponder来获取当前响应者对象的下一个响应者,过程如下:

1.首先看最佳响应者能否响应事件,如果他实现了touchesBegan/touchesMove/touchesEnded/touchesCancelled等方法,那么事件由他来处理,如果他不能处理事件,则会把事件传递给它的父视图,即它的nextResponder就会被指向它的父视图。

2.如果视图是控制器的根视图,nextResponder就是控制器对象,事件会被传递给控制器对象。

3.如果响应者都不能处理事件,那么事件在视图层次结构中一直传递到UIWindow对象,如果UIWindow对象也不处理,那么事件会被传递到UIApplication对象。

4.如果UIApplication也不能处理此事件,此事件被丢弃。

事件在响应者链的传递过程中,如果想要某个视图去处理事件,可以重写视图的touchesBegan方法,在重写方法中,如果事件处理完继续调用父类的super touchesBegan:withEvent: ,事件还会按照上述规则向下传递,如果不再调用父类的super touchesBegan:withEvent,事件则不再在响应者链中继续传递。

官网给出了图示及以下解释:应用程序中的响应者链:

如果UITextField不处理事件,则UIKit将该事件发送到UITextField的父视图对象,然后将其发送到窗口的根视图。从根视图开始,事件会被传递给视图控制器一致到UIWindow。如果UIWindow无法处理事件,则UIKit会将事件传递给该UIApplication对象,如果该对象是UIResponder响应者链的一个实例,并且还不是响应者链的一部分,则可能将该事件传递给应用程序委托。

总结

一次触摸事件产生后,事件被加入UIApplication管理的事件队列,UIApplication将事件从队列中取出并按照从下往上的顺序传递给视图层级结构中的视图,这形成了事件的传递链。事件的传递最终目的是为了找到事件的最佳响应者,而找到最佳响应者的关键方法是UIView对象的hitTest和pointInside方法。如果最佳响应者能够处理事件,那么传递终止于此,事件被处理完成,如果最佳响应者不处理事件,那么事件本身会沿着响应者链向上级视图传递,事件在响应者链中的传递是为了使响应者对事件做出响应,直到响应者链中所有响应者不能处理事件,事件最终被丢弃。


https://mp.weixin.qq.com/s/l0dofJrCRjOJwm-QvMQNyA

Axios 如何实现请求重试?

发布于:17天以前  |  66次阅读  |  详细内容 »

抖音 iOS 工程架构演进

发布于:22天以前  |  65次阅读  |  详细内容 »

【JS】625- Axios 如何缓存请求数据?

发布于:29天以前  |  87次阅读  |  详细内容 »

iOS中触摸事件的传递和响应分析

发布于:1月以前  |  101次阅读  |  详细内容 »

iOS中触摸事件的传递和响应分析

发布于:1月以前  |  114次阅读  |  详细内容 »

探索M1: 安装iOS版本微信/微信读书

发布于:1月以前  |  201次阅读  |  详细内容 »

iOS 稳定性问题治理:卡死崩溃监控原理及最佳实践

发布于:2月以前  |  167次阅读  |  详细内容 »

2021 给 iOS 开发者的一些建议

发布于:2月以前  |  283次阅读  |  详细内容 »

iOS 优化篇 - 启动优化之Clang插桩实现二进制重排

发布于:3月以前  |  280次阅读  |  详细内容 »

抖音品质建设 - iOS启动优化《实战篇》

发布于:3月以前  |  247次阅读  |  详细内容 »

iOS APP 图标版本化

在我们的项目开发过程中,需要频繁打包给测试人员去测试,有时候我们都不知道测试机上安装的版本是否是最新的,这样会造成很多不必要的麻烦和成本。因此我们需要将buildNumber以水印的方式打在APPIcon上,可以很直观的知道当前是哪一个版本。

发布于:3月以前  |  269次阅读  |  详细内容 »

如何实现一个HTTP请求库——axios源码阅读与分析

在前端开发过程中,我们经常会遇到需要发送异步请求的情况。而使用一个功能齐全,接口完善的HTTP请求库,能够在很大程度上减少我们的开发成本,提高我们的开发效率。

发布于:3月以前  |  261次阅读  |  详细内容 »

老司机 iOS 周报 #144 | 2021-01-14

发布于:3月以前  |  305次阅读  |  详细内容 »

快手,快影 iOS App反调试

发布于:4月以前  |  298次阅读  |  详细内容 »

优酷iOS插件化页面架构方法

随着业务不停地迭代,优酷 APP 用于分发视频资源的 UI 控件越写越多,也越来越复杂,并且同时相似相近的代码也非常多。

发布于:5月以前  |  390次阅读  |  详细内容 »

iOS中的内嵌汇编

写一篇在iOS上使用汇编的文章的想法在脑袋里面停留了很久了,但是迟迟没有动手。虽然早前在做启动耗时优化的工作中,也做过通过拦截objc_msgSend并插入汇编指令来统计方法调用耗时的工作,但也只仅此而已。刚好最近的时间项目在做安全加固,需要写更多的汇编来提高安全性(文章内汇编使用指令集为ARM64),也就有了本文

发布于:5月以前  |  417次阅读  |  详细内容 »

77.9K 的 Axios 项目有哪些值得借鉴的地方

Axios 是一个基于 Promise 的 HTTP 客户端,同时支持浏览器和 Node.js 环境。它是一个优秀的 HTTP 客户端,被广泛地应用在大量的 Web 项目中。

发布于:5月以前  |  383次阅读  |  详细内容 »

不会吧,这也行?iOS后台锁屏监听摇一摇

一般情况下,出于省电、权限、合理性等因素考虑,给人的感觉是很多奇怪的需求安卓可以实现,但是iOS就无法实现!今天要介绍的需求也有这种感觉,就是“当 APP 处于后台或锁屏状态时,依旧可以监听到摇一摇,进而触发某些功能,比如:语音播报”。

发布于:6月以前  |  529次阅读  |  详细内容 »

iOS 稳定性:App 被终止的原因

本次 session 主要内容如下: 介绍了后台应用终止的常见原因,并提供了一些优化建议 介绍了 MetricsKit 提供的在代码中获取诊断和性能数据的方法 介绍了 Xcode Metrics Ogranizer 提供的关于线上用户性能数据的可视化报告

发布于:6月以前  |  795次阅读  |  详细内容 »

优酷iOS插件化页面架构方法

随着业务不停地迭代,优酷 APP 用于分发视频资源的 UI 控件越写越多,也越来越复杂,并且同时相似相近的代码也非常多。

发布于:6月以前  |  523次阅读  |  详细内容 »

最多阅读

快速配置 Sign In with Apple 1年以前  |  4157次阅读
使用 GPUImage 实现一个简单相机 1年以前  |  2848次阅读
APP适配iOS11 2年以前  |  2759次阅读
开篇 关于iOS越狱开发 2年以前  |  2733次阅读
在越狱的iPhone设置上使用lldb调试 2年以前  |  2653次阅读
给数组NSMutableArray排序 2年以前  |  2636次阅读
App Store 审核指南[2017年最新版本] 2年以前  |  2584次阅读
所有iPhone设备尺寸汇总 2年以前  |  2509次阅读
UITableViewCell高亮效果实现 2年以前  |  2455次阅读
使用ssh访问越狱iPhone的两种方式 2年以前  |  2388次阅读
关于Xcode不能打印崩溃日志 2年以前  |  2276次阅读
使用ssh 访问越狱iPhone的两种方式 2年以前  |  2175次阅读
为对象添加一个释放时触发的block 2年以前  |  1938次阅读
UIDevice的简单使用 2年以前  |  1937次阅读
使用最高权限操作iPhone手机 2年以前  |  1918次阅读