上篇文章已经分析了Android的Touch事件分发。如果没看的建议先看一下。Android View的Touch事件分发。

接下来我们开始写几种场景,得出一个初步的执行顺序,然后我们按照这个顺序开始分析。

首先我们自定义一个ViewGroup和一个View,然后重写相关事件进行打印:

场景一:正常返回super,TouchView设置click和onTouchListener事件(onTouch返回false)

TouchViewGroup.png

TouchView.png

布局.png

TouchView设置事件.png

这时候我们点击一下TouchView,触发事件:

点击一下.png

可以看到触发的DOWN MOVE UP事件顺序都为:

ViewGroup.dispatchTouchEvent -> ViewGroup.onInterceptTouchEvent -> View.dispatchTouchEvent -> View.onTouch -> View.onTouchEven

只是在UP事件的时候最后多了一个click事件。

场景二:在场景一的基础上取消TouchView的onClick事件

TouchView取消click事件.png

这时候发现除了,执行的顺序变为了:

ViewGroup.dispatchTouchEvent -> ViewGroup.onInterceptTouchEvent -> View.dispatchTouchEvent -> View.onTouch -> View.onTouchEven->ViewGroup.onTouchEven

并且只有DOWN事件,其他事件就没有了。

场景三:在场景二的基础上TouchViewGroup的onInterceptTouchEvent里面返回true

这个时候就只有DOWN事件,并且顺序为:

ViewGroup.dispatchTouchEvent -> ViewGroup.onInterceptTouchEvent -> ViewGroup.onTouchEvent

接下来我们通过源码来分析:

首先从ViewGroup的dispatchTouchEvent入手

@Override

public boolean dispatchTouchEvent(MotionEvent ev) {

//...

boolean handled = false;

//...

//1.取消之前的手势

// Handle an initial down.

if (actionMasked == MotionEvent.ACTION_DOWN) {

// Throw away all previous state when starting a new touch gesture.

// The framework may have dropped the up or cancel event for the previous gesture

// due to an app switch, ANR, or some other state change.

cancelAndClearTouchTargets(ev);

resetTouchState();

}

//2.判断是否拦截

// Check for interception.

final boolean intercepted;

if (actionMasked == MotionEvent.ACTION_DOWN

|| mFirstTouchTarget != null) { //DOWN

//父类是否拦截 getParent().requestDisallowInterceptTouchEvent();来改变值

final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;

if (!disallowIntercept) {

intercepted = onInterceptTouchEvent(ev);

ev.setAction(action); // restore action in case it was changed

} else {

intercepted = false;

}

} else {

// There are no touch targets and this action is not an initial down

// so this view group continues to intercept touches.

intercepted = true;

}

//....

//3.0 如果是不取消不拦截为down,并且dispatchTransformedTouchEvent返回为true的时候会为 mFirstTouchTarget赋值

// Check for cancelation.

final boolean canceled = resetCancelNextUpFlag(this)

|| actionMasked == MotionEvent.ACTION_CANCEL;

// Update list of touch targets for pointer down, if needed.

final boolean split = (mGroupFlags & FLAG_SPLIT_MOTION_EVENTS) != 0;

TouchTarget newTouchTarget = null;

boolean alreadyDispatchedToNewTouchTarget = false;

//3.1 如果不取消并且不拦截的情况下,

if (!canceled && !intercepted) {

if (actionMasked == MotionEvent.ACTION_DOWN

|| (split && actionMasked == MotionEvent.ACTION_POINTER_DOWN)

|| actionMasked == MotionEvent.ACTION_HOVER_MOVE) {// 3.2 DOWN的时候

//...

if (newTouchTarget == null && childrenCount != 0) {

//...

final View[] children = mChildren;

for (int i = childrenCount - 1; i >= 0; i--) {//3.3 反序for循环,为了先拿到上层的view

//...

//3.4 拿到child

final View child = getAndVerifyPreorderedView(preorderedList, children, childIndex);

//...

//3.5 根据child给newTouchTarget赋值 DOWN的时候因为 mFirstTouchTarget==null 所以进不去 返回的是null

newTouchTarget = getTouchTarget(child);

}

//...

//3.6. 执行操作 是执行自己的dispatchTouchEvent还是child的dispatchTouchEvent

if (dispatchTransformedTouchEvent(ev, false, child, idBitsToAssign)) {

//...

//3.7 子View如果返回true添加一个newTouchTarget 并且为mFirstTouchTarget赋值

newTouchTarget = addTouchTarget(child, idBitsToAssign);

//....

}

}

}

}

//...

// Dispatch to touch targets.

if (mFirstTouchTarget == null) {//执行自身的dispatchTouchEvent

// No touch targets so treat this as an ordinary view.

handled = dispatchTransformedTouchEvent(ev, canceled, null,

TouchTarget.ALL_POINTER_IDS);

} else {// mFirstTouchTarget已经赋值

// Dispatch to touch targets, excluding the new touch target if we already

// dispatched to it. Cancel touch targets if necessary.

TouchTarget predecessor = null;

TouchTarget target = mFirstTouchTarget;

while (target != null) {

final TouchTarget next = target.next;

if (alreadyDispatchedToNewTouchTarget && target == newTouchTarget) {//执行完3.7操作的

handled = true;

} else {

final boolean cancelChild = resetCancelNextUpFlag(target.child)

|| intercepted;

if (dispatchTransformedTouchEvent(ev, cancelChild,

target.child, target.pointerIdBits)) {

handled = true;

}

if (cancelChild) {

if (predecessor == null) {

mFirstTouchTarget = next;

} else {

predecessor.next = next;

}

target.recycle();

target = next;

continue;

}

}

predecessor = target;

target = next;

}

}

return handled;

}

/**

* Cancels and clears all touch targets.

*/

private void cancelAndClearTouchTargets(MotionEvent event) {

if (mFirstTouchTarget != null) {

boolean syntheticEvent = false;

if (event == null) {

final long now = SystemClock.uptimeMillis();

event = MotionEvent.obtain(now, now,

MotionEvent.ACTION_CANCEL, 0.0f, 0.0f, 0);

event.setSource(InputDevice.SOURCE_TOUCHSCREEN);

syntheticEvent = true;

}

for (TouchTarget target = mFirstTouchTarget; target != null; target = target.next) {

resetCancelNextUpFlag(target.child);

dispatchTransformedTouchEvent(event, true, target.child, target.pointerIdBits);

}

clearTouchTargets();

if (syntheticEvent) {

event.recycle();

}

}

}

//清楚所有的TouchTarget

/**

* Clears all touch targets.

*/

private void clearTouchTargets() {

TouchTarget target = mFirstTouchTarget;

if (target != null) {

do {

TouchTarget next = target.next;

target.recycle();

target = next;

} while (target != null);

mFirstTouchTarget = null;

}

}

//根据childVie得到TouchTarget

/**

* Gets the touch target for specified child view.

* Returns null if not found.

*/

private TouchTarget getTouchTarget(@NonNull View child) {

// DOWN的时候因为 mFirstTouchTarget==null 所以进不去 返回的是null

for (TouchTarget target = mFirstTouchTarget; target != null; target = target.next) {

if (target.child == child) {

return target;

}

}

return null;

}

/**

* Transforms a motion event into the coordinate space of a particular child view,

* filters out irrelevant pointer ids, and overrides its action if necessary.

* If child is null, assumes the MotionEvent will be sent to this ViewGroup instead.

*/

private boolean dispatchTransformedTouchEvent(MotionEvent event, boolean cancel,

View child, int desiredPointerIdBits) {

//伪代码

final boolean handled;

if (child == null) {//执行View.dispatchTouchEvent 也就是自己的dispatchTouchEvent

handled = super.dispatchTouchEvent(event);

} else {//执行child的dispatchTouchEvent

handled = child.dispatchTouchEvent(event);

}

return handled;

}

//添加TouchTarget 并且给mFirstTouchTarget赋值

/**

* Adds a touch target for specified child to the beginning of the list.

* Assumes the target child is not already present.

*/

private TouchTarget addTouchTarget(@NonNull View child, int pointerIdBits) {

final TouchTarget target = TouchTarget.obtain(child, pointerIdBits);

target.next = mFirstTouchTarget;

mFirstTouchTarget = target;

return target;

}

当DOWN的时候,从注释和方法名可以看出,会调用cancelAndClearTouchTargets,然后在调用clearTouchTargets使mFirstTouchTarget = null用来废弃上一次的触摸手势。

接着判断父类需不需要拦截,先通过(mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0来判断,在这里可以通过getParent().requestDisallowInterceptTouchEvent(boolean disallowIntercept)来改变值,如果上面为判断为false再通过onInterceptTouchEvent的返回值来确定,这个函数默认情况下返回false。

检测是否为取消事件,如果不是取消、不拦截并且为 DOWN事件的时候,就会对childView一个反序的for循环来遍历,并且执行dispatchTransformedTouchEvent操作,这个操作用来执行dispatchTouchEvent,如果childView是null的话将执行View.dispatchTouchEvent,也就是自己的dispatchTouchEvent,反之执行childView的dispatchTouchEvent,如果执行dispatchTransformedTouchEvent返回的值是true那么将会调用addTouchTarget()为这个childView生成一个TouchTarget并且执行mFirstTouchTarget = target将之赋值于mFirstTouchTarget ,然后跳出for循环遍历。

判断操作,首先判断mFirstTouchTarget是否为null,如果是DOWN事件,不拦截不取消并且dispatchTransformedTouchEvent返回了true,那么将会不进入这个判断,如果不是,那么将会在这执行自身的dispatchTouchEvent函数并且将返回值赋于handled返回。进入else语句,在里面将其mFirstTouchTarget进行next遍历,里面的if语句则是DOWN事件下的dispatchTransformedTouchEvent返回true的情况,直接将其赋值,然后返回,里面的else语句则是,调用dispatchTransformedTouchEvent,然后将其返回值返回。

到这里,ViewGroups事件分发源码的流程就分析了,我们根据这个来说说上面的场景。

场景一:我们在TouchViewGroup的dispatchTouchEvent正常返回super,DOWN事件先触发TouchViewGroup的dispatchTouchEvent,然后就执行onInterceptTouchEvent是否拦截,onInterceptTouchEvent返回的是super,也就是false,所以就会通过dispatchTransformedTouchEvent来执行TouchView的dispatchTouchEvent,后面就是View的Touch事件分发了,View流程将会按照dispatchTouchEvent->onTouchListener - > onTouchEvent的顺序执行,因为设置了点击事件,所以在这里就返回了true,这个时候就会通过addTouchTarget()给mFirstTouchTarget赋值,下面就直接返回了true。然后在MOVE和UP事件的时候,也是首先执行dispatchTouchEvent,调用super然后调用onInterceptTouchEvent询问是否拦截,还是false,但是这里因为不是DOWN事件,所以就不会进入判断对其childView反遍历,因为在DOWN的时候mFirstTouchTarget赋值了,所以这时候进入第4步的else语句里面,这时候就对其遍历执行dispatchTransformedTouchEvent,也就是dispatchTouchEvent,然后将其返回。

场景2:我们取消了点击事件,那么在DOWN的时候就不会给mFirstTouchTarget赋值,这个时候将会进入第4步的if判断里面,直接调用dispatchTransformedTouchEvent,所以事件就不会有拦截,最终返回false,所以后续将不会接受到任何事件

场景3:我们在TouchViewGroup的时候是在onInterceptTouchEvent返回true,所以我们intercepted=true,这时候就不会给mFirstTouchTarget赋值,这个时候就调用自身的dispatchTransformedTouchEvent,同样的返回false,后续将不会接受到事件。

通过源码的角度我们也知道了为什么会这么执行,初步有点模糊,我们需要通过项目慢慢的来完善对它的认知。希望对大家有所帮助。

android viewgroup点击变色,Android ViewGroup事件分发相关推荐

  1. android 字体点击变色,Android TextView 中实现部分文字变色以及点击事件

    首先要想实现文字变色以及点击,都需要使用到SpannableStringBuilder,实例化该类也很简单,只需将你想要处理的字符串当做参数 SpannableStringBuilder spanna ...

  2. android按钮点击变化,Android实现按钮点击效果(第一次点击变色,第二次恢复)...

    1.首先创建一个按钮 android:id="@+id/click" android:layout_width="fill_parent" android:la ...

  3. Android系统分析之带着问题看事件分发机制

    Android 触摸事件分发机制? Android系统分析之带着问题看事件分发机制 一 事件分发机制 1 什么是事件分发机制? 1.1 什么是事件? 答:当用户触摸屏幕时,每一次的点击,按下,移动,抬 ...

  4. Android View系列(二):事件分发机制源码解析

    概述 在介绍点击事件规则之前,我们需要知道我们分析的是MotionEvent,即点击事件,所谓的事件分发就是对MotionEvent事件的分发过程,即当一个MotionEvent生成以后,系统需要把这 ...

  5. android 模拟手指点击,『Android Tip』-- 模拟手势操作

    平时 Android 开发中总会遇到奇葩的功能或者需求,这里做个记录和积累,以便后面开发过程中遇到类似的问题,可以快速的解决.Android tips 前言 这个版本终于快结束了,历时一个月的时间,这 ...

  6. android view 点击变暗,Android应用开发Android ImageView点击变暗效果

    本文将带你了解Android应用开发Android ImageView点击变暗效果,希望本文对大家学Android有所帮助. < 自定义ImageView: 在ImageView中setPres ...

  7. android listview 点击获取焦点,android – ListView项目焦点行为

    我正在尝试创建一个EditText项目的ListView(使用自定义CursorAdapter),这样EditTexts最初看起来是不可编辑的,并且在长时间单击后变得可编辑.然后,用户将编辑EditT ...

  8. android radiobutton 点击效果,Android RadioButton 的点击效果

    效果图 控件 android:divider="@drawable/wosim_rg_divider" android:showDividers="middle" ...

  9. android 模拟点击localinstrumentation,Android Instrumentation模拟鼠标点击事件

    看了几遍网上的博客一直没有 模拟出鼠标点击事件和按钮事件,后来抱着试试态度再重试的时候终于有所斩获.下面把具体的情况记录一下: 首先我们必须了解类 Instrumentation: Instrumen ...

最新文章

  1. Error in apply(df$var1, 2, mean) : dim(X) must have a positive length
  2. 将base64编码图片上传到七牛云
  3. 关闭串口_USART串口通信,DMA方式,一分钟从入门到大师
  4. numpy添加元素_科研速递 | 花费15年众望所归!NumPy论文终登上Nature!
  5. 分享一些user-agent(移动端和PC端都有)
  6. 找第一个只出现一次的字符_剑指offer 字符流中第一个只出现一次的字符
  7. Flume安装(单节点)
  8. 【转】关于Class.forName(“com.mysql.jdbc.Driver”)
  9. 对字符串进行加密解密
  10. 爬取表格类网站数据并保存为excel文件
  11. 巧用MacOS的勿扰模式,解决广告弹窗
  12. springboot过滤器配置
  13. XML注释内容中不能出现“--”
  14. 洛谷 P1359 租用游艇(Floyd, Dijkstra,SPFA)
  15. 文件的打开方式怎么用计算机,电脑怎样修改文件默认打开方式
  16. 电脑上m4a怎么转换成mp3
  17. 阿里云域名注册赠送的阿里云企业邮箱使用方法
  18. Hybrid Dilated Convolution学习笔记
  19. 教你作一份高水准的简历
  20. 我的世界(18)-精英怪物(InfernalMobs插件)

热门文章

  1. 使用Cloudformation集成Spring Boot和EC2
  2. Java 8:在2分钟内将智能流与数据库一起使用
  3. Jibx Jersey2集成
  4. Spring Data JDBC通用DAO实现–迄今为止最轻量的ORM
  5. PostgreSQL PL / java简介
  6. 无需复杂插件即可从Eclipse启动和调试Tomcat
  7. JavaFX移动应用程序最佳实践,第2部分
  8. java 哈希一致算法_一致哈希算法Java实现
  9. 字符集和字符编码的学习
  10. 如何使用SSH客户端远程操作linux系统,并启动、关闭tomcat和查看后台日志