Flutter 指针事件原理&点击穿透_flutter pointer.fromfunction-程序员宅基地

技术标签: 程序员  flutter  android  

隔离的这14天,慢慢的研究了Flutter的指针事件,在这个过程中,又重新梳理了一下Element和Render Tree的形成过程。这篇文章,主要对指针事件在Fluter中如何下发到各个组件的过程进行梳理。(指针是指针,手势是手势,手势是指针事件的某种行为,且只有一个胜者,这点要区分清楚。)

好像要一只dash啊。

好的,进入正题。当你点击了屏幕,Flutter做了什么呢?

过程解析

通过调用栈,我们逐步分析过程。

屏幕截图 2021-09-22 203207.png

1.从平台处获得指针事件数据并分发

指针数据被包装成Dart的ByteData类型,调用PlatformDispatcher_dispatchPointerDataPacket方法,至于指针数据怎么被平台包装,内容是什么,这就涉及到各个平台处理指针的方式,掘金由许多相关文章。(作者目前不懂任何原生知识)

image.png

2.在GestureBinding和中处理得到的指针数据

经过了几层看不懂的调用,指针数据走到了这个地方,并且以队列的形式被处理。(这里又有window,又有lock,暂时不管)。

image.png

注意到,这里的packet是ui.PointerDataPacket类型,该类内部仅有一个final List<PointerData>的data成员。

image.png

既然是队列,那么就是循环出列,逐个处理事件。这个过程由_flushPointerEventQueue完成。

image.png

handlePointerEvent方法,如其名,这里传过来的数据已经成为了PointerEvent类型。(resanlingEnabled默认为false,官方文档的解释是通过这只这个选项,对于设备的指针采样率屏幕刷新率有某种关系的,可以让指针更丝滑)

image.png

3.在GestureBinding和RendererBinding中进行hitTest

_handlePointerEventImmediately方法被handlePointerEvent(上面那个家伙)调用。 这个方法是个重量级的家伙。

void _handlePointerEventImmediately(PointerEvent event) {
  HitTestResult? hitTestResult;
  if (event is PointerDownEvent || event is PointerSignalEvent || event is PointerHoverEvent) {
    //省略...
    hitTestResult = HitTestResult();
    hitTest(hitTestResult, event.position);   //看这里!!!!!!!!!!!
    //省略...
    }
    dispatchEvent(event, hitTestResult);
  }

}
复制代码

到这里,先缓一缓,也许上面我写的不好看不懂也没关系,你只需要记住一件事,我们已经拿到了类型为PointerEvent的指针事件的数据event

(PointerEvent有多个子类,PointerDownEventPointerMoveEventPointerUpEvent等等,对应点击移动抬起)

核心内容开始

这里以PointerDownEvent举例,这个事件为用户点击屏幕后产生的。

为什么用户要点击?我猜他在某种APP内发现了一张涩图想点进去看看。图片肯定是个RenderObject(不然你能看到个**),那写代码的怎么知道用户点的是哪张呢?

Flutter带Hit开头的接口帮我们做这件事,和RenderObject相关的有三个接口。

1. HitTestablehitTest方法,让这个RenderObject能点,什么是能点?稍后的HitTestResult就会告诉你。

2. HitTestDispatcherdispatchEvent方法,嗯,能发事件。

3. HitTestTargethandleEvent方法,RenderObject能被点了,那事件你处理不处理,怎么处理,就是这个方法的内容了。

高能来了

RenderViewhitTest进行递归,跟据点击指针事件eventposition,调用childhitTestRenderView是RenderTree的根,怎么来的可以看看Binding的相关内容。

image.png

这里插一则,GestureBindinghitTestRendererBinding也有,从runApp方法可以看到调用内容,super.hitTest对应GestureBindinghitTest,截图对应的是RendererBinding。(不影响阅读后面的内容)

递归调用内容解析

RenderTree从RenderView开始hitTest。大多数情况下,我们创建的都是RenderBox,这个盒模型,有长和宽等大小信息,使用笛卡尔坐标系。有的有单个child,或者双链表式的children。面对这多种情形,不同的组件有不同的点击测试内容。

这里通过StackContainer组件,让大家理解下这个递归过程。

Stack(
  children: [
    GestureDetector(
        behavior: HitTestBehavior.translucent,
        onTap: (){print("blue");},
        child: Container(width: 300,height: 300,color:Colors.blue)),
    GestureDetector(
        behavior: HitTestBehavior.translucent,
        onTap: (){print("red");},
        child: Container(width: 150,height: 150,color:Colors.red))
  ],
),
复制代码

image.png

某些乱七八糟的BLOG说设置GestureDetectorbehavior就能实现点击穿透,然而点击红色方块控制台只输出red。

这是为什么呢?答案是和hitTest有关。

Stack对应的是RenderStack,是一个双链表child模型(由ContainerRenderObjectMixin实现)。其hitTestRenderBox的方法,原汁原味。

如果这个Box包含点击点,通过hitTestChildren先对children逐个进行hitTest,前者不中则再通过hitTestSelf自己进行hitTest。只要中了,就把自己添加进result

result储存的是所有通过的测试的RenderBox,这些都会接收到指针事件。

image.png

RenderStackhitTestChildren

hitTest过程可以解释为从lastChild开始,向前进行点击测试,直到有一个child通过,addWithPaintOffset的内部会添加进result,然后返回true

为什么从lastChild开始?因为是栈顶对应的RenderBox,这样就保证了上面的盖住了下方的,使得一般情况下的点击无法穿透。

image.png

image.png

hitTestSelf默认返回false,子类可以根据需要重写。(GestureDetectorRenderPointerListenerListenerRenderBoxWithHitTestBehavior等有详细的内容,之后的点击穿透会讲)

所以结论是,在HitTestChildren中,红色方块的RenderBox被添加进了HitTestResult中,此时就跳出循环,递归回调,所以蓝色方框得不到指针信息。

hitTest总结

这是一个自上而下递归的过程,内部主要由hitTestChildrenhitTestSelf实现。点击处的坐标在RenderBox的内部是能够进行hitTest的前提,但是通不通过hitTest取决于组件内部自己是如何实现hitTest,能否接收到指针事件取决于是否把RenderBox添加到HitTestResult中。

好奇result的内容,在_handPointEventImmediately中打印result即可。

dispatchEvent分发通知

把获得的HitTestResult通过内部path遍历,调用各个RenderObjecthandleEvent。其实内部还有pointRoute等内容,暂时没研究。

image.png

在插播一条无关消息,GestureBinding这个方法下面的handleEvent就是手势竞技场的内容。(手势是手势的事情,指针不管手势的事)

总结

所以平台的指针事件下发需经历如下3个过程。

  1. 包装成Flutter能看得懂的指针数据

  2. 挑选出需要响应事件的RenderObject

  3. 分发事件执行RenderObjecthandleEvent

更多Android知识,扫码即可了解

<img src="https://hnxx.oss-cn-shanghai.aliyuncs.com/official/1704935888404.jpg?t=0.014764499839815759" style="margin: auto" />

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/Code1994/article/details/120570418

智能推荐

【keras-DeepLearning_Models】_obtain_input_shape() got an unexpected keyword argument 'include_top'_shape为什么报错-程序员宅基地

文章浏览阅读1w次,点赞7次,收藏11次。最近想跑一些主流的网络感受感受。从github上找到了 deep-learning-models 提供的几个模型,包括:inception-v2, inception-v3, resnet50, vgg16, vgg19 等等。这些代码都是基于 keras 框架,正好我最近有在学 tensorflow 和 keras,所以很想跑跑这些代码。心动不如行动,准备工作都做得差不多了,准备开始跑代码。此时,出现了一些常见的问题,也正好借此机会整理下来。_shape为什么报错

windowsXP下安装Qt4.8_xp的系统装qt-程序员宅基地

文章浏览阅读1.5k次。参考1:http://blog.csdn.net/areswhy/article/details/8697527参考2:http://www.cnblogs.com/emouse/archive/2013/01/29/2881721.html(windows下Qt安装部分)前记:一两个月以后就要找工作了,本来想学学embedded-linux Qt,无奈电脑老旧,用的虚拟机装的ubuntu_xp的系统装qt

KNN Python-Iris Data Set (鸢尾属植物数据集)_iris_data.mat-程序员宅基地

文章浏览阅读2.2k次。Iris数据集概况Iris Data Set(鸢尾属植物数据集)是我现在接触到的历史最悠久的数据集,它首次出现在著名的英国统计学家和生物学家Ronald Fisher 1936年的论文《The use of multiple measurements in taxonomic problems》中,被用来介绍线性判别式分析。在这个数据集中,包括了三类不同的鸢尾属植物:Iris Setosa,I..._iris_data.mat

解决React在安装antd之后出现的Can't resolve './locale'或者浏览器显示Cannot find module './locale’问题的两种方案_react脚手架 can't resolve './const-程序员宅基地

文章浏览阅读1.1w次,点赞4次,收藏3次。React在安装antd之后出现的Can’t resolve './locale’或者浏览器显示Cannot find module './locale’问题,是因为moment的版本有问题,而react默认使用了最新的moment,但是在[email protected]中是没有问题的。1.第一种解决方法解决方案就是配置webpack的alias,将所有的 moment 路径引用导入到 [email protected]操作步骤安装moment 依赖 npm install [email protected]_react脚手架 can't resolve './const

Redis之下载与安装(二)_(二)redis下载和安装-程序员宅基地

文章浏览阅读161次。其实Redis并不正式支持windows版本,官网那个5.0.x稳定版其实是linux版本_(二)redis下载和安装

AD10 PCB文件统一设置字体大小(默认字体大小)_ad10统一修改字符大小-程序员宅基地

文章浏览阅读4.3w次,点赞10次,收藏23次。这个网上给我的提示(http://zhidao.baidu.com/link?url=UNxy0GoaU7jj0QRgCikKIdHIrE7C-FOiojG-5nE6a_QBqXVuYdublOROizQyNRtfNudH53WZQrGj6Dgv_sXcr_):默认值设置:Tools_Preference_PCB Editor_Defaults,选Component点Edit Values.._ad10统一修改字符大小

随便推点

zencart1.55stripe信用卡内嵌支付获取卡号_zencart 回调方法checkout_process 提交参数-程序员宅基地

文章浏览阅读7.4k次。zencart1.55内嵌支付获取卡号_zencart 回调方法checkout_process 提交参数

pl/sql---约束_plsql里面约束在哪显示-程序员宅基地

文章浏览阅读474次。多表连接一.什么是约束二,注意三,表级约束和列级约束命令行:四,定义约束五,unique约束命令语句:六,主键约束命令语句01:命令语句02:七,foreign key 约束命令语句:七,foreign key 约束的关键字八,check 约束命令行:..._plsql里面约束在哪显示

javax.net.ssl.SSLException: Received fatal alert: internal_error 解决-程序员宅基地

文章浏览阅读1w次,点赞2次,收藏2次。javax.net.ssl.SSLException: Received fatal alert: internal_error_javax.net.ssl.sslexception: received fatal alert: internal_error

Oracle & 神通数据库 清空所有表数据操作_神通数据库清除数据的脚本-程序员宅基地

文章浏览阅读1.3k次,点赞2次,收藏2次。1、拼接处truncate所有表的语句select 'truncate table '||table_name||';' from user_tables;2、Ctrl+A全选,粘贴至命令行,执行Over._神通数据库清除数据的脚本

切图常说的@1X@2X@3X是什么意思?_web1x 2x-程序员宅基地

文章浏览阅读1w次。苹果IOS程序开发不同分辨率的设备统一为一个尺寸而标记的。@3X就是@1X分辨率的3倍。如图,iPad2 是768 x 1024,iPad Retina 是1536 x 2048,开发时都按 768 * 1024 操作。但实际上两者有一倍差异。为了达到最佳效果,使用的图片大小不一样。这时候就用同一个名称,但 Retina 的图加上 @2x 后缀。系统加载图片时,在 iPad2 上会加载 @_web1x 2x

element-ui el-input / el-select输入框的非空校验_vue2对el-select和el-input如何判断是否为空-程序员宅基地

文章浏览阅读1.7w次,点赞4次,收藏8次。一、之前写的实现直接js判断就可以.prevent .stop 是阻止继续冒泡 不一样的见 官方说明 1https://cn.vuejs.org/v2/cookbook/form-validation.html#%E4%BD%BF%E7%94%A8%E8%87%AA%E5%AE%9A%E4%B9%89%E6%A0%A1%E9%AA%8C2 自定义 (我没看)https://..._vue2对el-select和el-input如何判断是否为空