当前位置: 首页 > news >正文

Android7 Input(九)View 建立Input Pipeline

概述

上一个章节,我们讲解了App通过注册InputEventReceiver获取从Input服务上报的输入事件的流程,在讲述App处理具体输入事件之前,我们讲述App处理输入事件的模型,也就是本章要讲述的InputState Pipeline。

源码路径

frameworks/base/core/java/android/view/ViewRootImpl.java

Input事件处理模型

View Input事件处理的Pipeline模型建立在InputStage这个抽象接口基础之上,我们这里先讲述InputStage的实现,我们只列出InputStage的核心接口实现,如下所示:

abstract class InputStage {private final InputStage mNext;protected static final int FORWARD = 0;protected static final int FINISH_HANDLED = 1;protected static final int FINISH_NOT_HANDLED = 2;public InputStage(InputStage next) {mNext = next;}public final void deliver(QueuedInputEvent q) {if ((q.mFlags & QueuedInputEvent.FLAG_FINISHED) != 0) {forward(q);} else if (shouldDropInputEvent(q)) {finish(q, false);} else {apply(q, onProcess(q));}}protected void finish(QueuedInputEvent q, boolean handled) {q.mFlags |= QueuedInputEvent.FLAG_FINISHED;if (handled) {q.mFlags |= QueuedInputEvent.FLAG_FINISHED_HANDLED;}forward(q);}protected void forward(QueuedInputEvent q) {onDeliverToNext(q);}protected void apply(QueuedInputEvent q, int result) {if (result == FORWARD) {forward(q);} else if (result == FINISH_HANDLED) {finish(q, true);} else if (result == FINISH_NOT_HANDLED) {finish(q, false);} else {throw new IllegalArgumentException("Invalid result: " + result);}}protected int onProcess(QueuedInputEvent q) {return FORWARD;}protected void onDeliverToNext(QueuedInputEvent q) {if (DEBUG_INPUT_STAGES) {Log.v(mTag, "Done with " + getClass().getSimpleName() + ". " + q);}if (mNext != null) {mNext.deliver(q);} else {finishInputEvent(q);}}}

1、deliver的实现,因为他决定input事件处理方式方法实现如下:

 public final void deliver(QueuedInputEvent q) {if ((q.mFlags & QueuedInputEvent.FLAG_FINISHED) != 0) {forward(q);} else if (shouldDropInputEvent(q)) {finish(q, false);} else {apply(q, onProcess(q));}}

deliver的核心逻辑,如下所示:

a、当前input事件直接被转发到下一级的InputStage进行处理 forward;

b、如果input事件在当前InputStage中被丢弃了,直接设置input事件被处理完成标记,并转发给下一级的InputStage;

c、当前InputStage进行处理调用process方法,然后根据处理的结果,决定input输入事件,是否向下一级的InputStage进行处理;

2、finish的实现非常简单,将事件打上FINISHED标签,然后转发;

 protected void finish(QueuedInputEvent q, boolean handled) {q.mFlags |= QueuedInputEvent.FLAG_FINISHED;if (handled) {q.mFlags |= QueuedInputEvent.FLAG_FINISHED_HANDLED;}// 当前InputStage不再处理,直接转发forward(q);}

3、forward 转发给下一级的InputStage处理

protected void forward(QueuedInputEvent q) {onDeliverToNext(q);}

4、apply 根据result结果,其本质也就是调用onProcess的返回结果,然后决定input事件下一步的处理方式。

protected void apply(QueuedInputEvent q, int result) {if (result == FORWARD) {forward(q);} else if (result == FINISH_HANDLED) {finish(q, true);} else if (result == FINISH_NOT_HANDLED) {finish(q, false);} else {throw new IllegalArgumentException("Invalid result: " + result);}
}

5、onProcess 的默认行为是转发。子类可以覆盖其实现,决定input事件的处理方式;

 protected int onProcess(QueuedInputEvent q) {return FORWARD;}

6、onDeliverToNext 直接转发给下一级的InputStage进行处理

 protected void onDeliverToNext(QueuedInputEvent q) {if (DEBUG_INPUT_STAGES) {Log.v(mTag, "Done with " + getClass().getSimpleName() + ". " + q);}if (mNext != null) {mNext.deliver(q);} else {finishInputEvent(q);}}

7、finishInputEvent 结束事件的处理流程。然后直接将处理结果返回给InputEventReceiver处理。

private void finishInputEvent(QueuedInputEvent q) {Trace.asyncTraceEnd(Trace.TRACE_TAG_VIEW, "deliverInputEvent",q.mEvent.getSequenceNumber());if (q.mReceiver != null) {boolean handled = (q.mFlags & QueuedInputEvent.FLAG_FINISHED_HANDLED) != 0;q.mReceiver.finishInputEvent(q.mEvent, handled);} else {q.mEvent.recycleIfNeededAfterDispatch();}recycleQueuedInputEvent(q);}

Input Pipeline的建立

Input Pipeline的建立,是在ViewRootImpl中的setView方法中,如下所示:

void setView(View view, WindowManager.LayoutParams attrs, View panelParentView) {......// Set up the input pipeline.CharSequence counterSuffix = attrs.getTitle();mSyntheticInputStage = new SyntheticInputStage();InputStage viewPostImeStage = new ViewPostImeInputStage(mSyntheticInputStage);InputStage nativePostImeStage = new NativePostImeInputStage(viewPostImeStage,"aq:native-post-ime:" + counterSuffix);InputStage earlyPostImeStage = new EarlyPostImeInputStage(nativePostImeStage);InputStage imeStage = new ImeInputStage(earlyPostImeStage,"aq:ime:" + counterSuffix);InputStage viewPreImeStage = new ViewPreImeInputStage(imeStage);InputStage nativePreImeStage = new NativePreImeInputStage(viewPreImeStage,"aq:native-pre-ime:" + counterSuffix);mFirstInputStage = nativePreImeStage;mFirstPostImeInputStage = earlyPostImeStage;mPendingInputEventQueueLengthCounterName = "aq:pending:" + counterSuffix;......
}

整个流水线的连接过程我们不再描述,最终建立起来了如下所示的流水线结构,流水从左向右进行执行,流水的起点由mFirstInputStage或者mFirstPostImeInputStage开始。

总结

本文描述了View处理input事件的Pipeline模型的实现,下一个章节讲述App处理input事件的具体过程。

相关文章:

  • 15 dart类(get,set,静态,继承,抽象,接口,混入)
  • Gartner报告解读《Technical Professionals Need to Track 5 ImportantLLM Developments》
  • 论文审稿之我对SCI写作的思考
  • CSS之元素定位
  • 批量获取电商商品数据的解决方案|API接口自动化商品采集|item_get 接口详解
  • 动态规划算法:字符串类问题(2)公共串
  • 【电子通识】FPC连接器组成部分与不良案例术语
  • Day02
  • 嵌入式学习笔记——day25
  • 英伟达破局1000 Token/秒!Llama 4以光速重塑AI推理边界
  • 【深度学习】1. 感知器,MLP, 梯度下降,激活函数,反向传播,链式法则
  • 微信小程序 --三剑客
  • STM32的内部FLASH
  • 「OC」源码学习——KVO底层原理探究
  • 30字速成Docker安装与配置指南
  • urdf文件和DH模型参数是一一对应的吗??
  • PySide6 GUI 学习笔记——常用类及控件使用方法(常用类图标QIcon)
  • ​《分布式年夜》
  • PTA刷题笔记(纠细节 有详解)
  • 【OCCT+ImGUI系列】010-BRepMesh-网格化IncrementalMesh
  • 网站兼容ie7/视频剪辑培训
  • 网站界面设计如何实现功能美与形式美的统一/百度热度榜搜索趋势
  • 关于网站建设方案/网站关键词怎么写
  • 做门户网站需要学什么知识/东莞专业网站推广工具
  • 有没有个人做试卷网站的/佛山市人民政府门户网站
  • 佛山做外贸网站信息/广州网络推广外包平台