android viewgroup点击变色,Android ViewGroup事件分发

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

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

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

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

95145e4f0642

TouchViewGroup.png

95145e4f0642

TouchView.png

95145e4f0642

布局.png

95145e4f0642

TouchView设置事件.png

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

95145e4f0642

点击一下.png

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

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

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

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

95145e4f0642

95145e4f0642

TouchView取消click事件.png

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

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

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

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

95145e4f0642

95145e4f0642

这个时候就只有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,后续将不会接受到事件。

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/340322.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

qq数据泄露_真良心,腾讯这个app竟然能查账号泄露

最近有朋友都在分享腾讯手机管家,纷纷表示这是一个良心app,能查到自己账号泄露。仔细想一下,这么多年下来确实忘了自己都在什么平台或网站注册过账号,在黑客泛滥的今天,黑客很容易从一些平台窃取数据库,拿到…

vue 如何获取图片的原图尺寸_阳台洗衣机组合柜如何设计|尺寸规范|案例图片...

对于阳台装洗衣机来说,不只是简单的装修,还需要我们考虑水电、尺寸以及美观实用等问题,以免后期出现问题的时候会更加麻烦。接下来深圳装修网小编就为你们带来阳台洗衣机组合柜的案例赏析,以及装修的尺寸规范等内容,一…

print2flashwindows7旗舰版下载哪一个_JUJUMAO_MSDN原版 win 7 二合一 旗舰版32位 64位原版ISO镜像...

文件: F:\JUJUMAO_msdn_Win7_ultimate_x86_x64.iso大小:4.73G(5082120192 字节) MD5: AA4C7E80C52AC0DEDC757EF86CF8057BSHA1: 66505AD9424ED2D2B0DEDE7067917B708A67C7DDCRC32: 0E2FAB2C高速下载地址:https://jujumao.cowtransfer.com/s/3220a80131744fJUJUMAO_MSD…

lambda ::表达式_Lambda表达式和流API:基本示例

lambda ::表达式这篇博客文章包含基本Lambda表达式和Stream API示例的列表,我在2014年6月在Java用户组Politechnica Gedanensis (格但斯克技术大学)和Goyello的实时编码演示中使用了这些示例。 Lambda表达式 句法 最常见的示例: …

android readonly file system,安卓ROOT权限下“Read-only file sytem”解决办法

今天用安卓模拟器:BlueStacks,打开apk终端模拟器:Terminal,在shell操作命令的时候提示“Read-only file sytem”:第一种方法:在 Android 系统中,我们通过 adb 登录到 shell 进行操作时&#xff…

客制化键盘编程_客制化键盘劝退指南

客制化键盘劝退指南最近总感觉mac book pro的键盘不太好用,所以把家里用的杜伽K320拿到公司用了。这把键盘,樱桃的银轴,红轴的压力克数,更小的键程,用起来还是不错的。如此一来,家里的台式机没有键盘了&…

HTML5怎样设置站点,我是怎样让网站用上HTML5 Manifest

Manifest是用来做离线页面的,即使断网也能正常打开页面,用起来简单,但是在实际使用中存在以下问题:(1)如何自动缓存所有的页面的资源?因为manifest不能使用*通配符进行cache(2)如果网站资源更新,怎么让mani…

wallpaper怎么设置锁屏_Apple ID密码忘了怎么找回?丨如何让面容和指纹解锁立马失效?...

忘记Apple ID这种事肯定不少果粉都经历过,像小编也是如此,因为密码都比较复杂,如果太久没输入过密码,久而久之就忘了。哪一天突然需要用的时候,怎么想也想不起来。想不起来的话我们就别想了,直接重置密码就…

在Java中将时间单位转换为持续时间

java.util.concurrent.TimeUnit以给定的粒度单位表示Java中的持续时间,并提供了跨单位转换的实用方法。 java.util.concurrent.TimeUnit最早是在Java早期(1.5)引入的,但自那时以来已经被扩展了好几次。 在此博客文章中&#xff0c…

ajax contenttype详解_$.ajax中contentType: “application/json” 的用法详解

具体内容如下所示:$.ajax({type: httpMethod,cache:false,async:false,contentType: "application/json; charsetutf-8",dataType: "json",//返回值类型url: pathurl,data:jsonData,success: function(data){var resultData 返回码data.status…

在text html模版中写js,Rails3使用text/html内容类型而不是text/javascript呈现js.erb模板...

我正在使用3.0.0.beta3构建一个新的应用程序.我只是尝试将js.erb模板呈现给Ajax请求以执行以下操作(在publications_controller.rb中):def get_pubmed_dataentry Bio::PubMed.query(params[:pmid])# searches PubMed and get entrypublication Bio::MEDLINE.new(entry) # cre…

jq监听子元素被点击_vue开发app点击字母展示地区列表(兄弟组件之间联动)

下图这种地区搜索方式在很多app中都很常见,今天就使用vue框架中的 better-scroll 第三方包来实现页面滚动和点击侧边栏字母该字母开头的地区列表置顶功能。1、A子组件通过使用 this.$emit(事件名字,事件携带内容) 向外触发事件首先,在组件每个…

e-mobile帐号状态存在异常_一文掌握异常检测的实用方法 | 技术实践

作者 | Vegard Flovik译者 | Tianyu责编 | Jane出品 | AI科技大本营(ID: rgznai100)【导读】今天这篇文章会向大家介绍几个有关机器学习和统计分析的技术和应用,并展示如何使用这些方法解决一些具体的异常检测和状态监控实例。相信对一些开发…

用Spring组成自定义注释

Java批注在2004年随Java 5一起引入,是一种将元数据添加到Java源代码中的方法。 如今,许多主要框架(如Spring或Hibernate)都严重依赖注释。 在本文中,我们将介绍一个非常有用的Spring功能,该功能允许我们基…

单片机sleep函数的头文件_单片机代码模块化设计思想浅谈

前言:前段时间分享的文章【单片机裸机代码框架设计思路】,很多读者给我留言,觉得很不错,对于初学者而言,这是一个进阶的技巧,对于我而言,这是对自己总结和表达能力的一个提升。本文章我们再谈谈…

html中显示数据库中的一条数据,如何使用html表显示数据库中的数据

我正在尝试在HTML表中显示数据库中的数据。主要问题是:该表未出现。IdStringArray// Mostrar dadosecho "{$databaseName visteon;$pdo new Pdo(mysql:host127.0.0.1;dbname . $databaseName, root, );$result $pdo->query(SHOW TABLES FROM . $databas…

jquery 字符串查找_JQuery、Vue等考点

一. 写出下面程序的运行结果for结果:1秒后一下子打印出5个5。当循环完成时才会轮到setTimeout异步执行其回调函数function,此时i已经变成5,故5个console.log(i)里的i全使用的是5。易错点:千万别写成“打印5个4”啊!暴风…

html网页效果分析,熟手的html编写风格与原因分析_HTML/Xhtml_网页制作

一、导航:无序列表 对 其它标签元素用最常用的“无序列表“来写导航的理由是显而易见的,它代表一列链接,这本身就有足够的理由应选择列表标签。但需要移除list列表的默认样式,以使其更有意义。另一个好处可能超出你的想象&#xf…

qgis 图片_QGIS入门教程公告!!!

从本周开始,我将每周日更新一期QGIS入门教程视频,带讲解,估计这也是很多朋友所希望的。操作中用到的数据下载链接我会放在每一期的视频下方。以下是入门课程的目录:1.开启你的QGIS之旅——制作你的第一张地图2.属性表基础——字段…

v-if 表单验证_避免许多if块进行验证检查

v-if 表单验证在某些情况下,我们需要先验证输入数据,然后再将其发送到业务逻辑层进行处理,计算等。这种验证在大多数情况下是孤立进行的,或者可能包括与外部数据或其他输入的一些交叉检查。 看下面的示例,该示例验证用…