综述

在上篇文章Android中的事件分发机制(上)——ViewGroup的事件分发中,对ViewGroup的事件分发进行了详细的分析。在文章的最后ViewGroup的dispatchTouchEvent方法调用dispatchTransformedTouchEvent方法成功将事件传递给ViewGroup的子View。并交由子View进行处理。那么现在就来分析一下子View接收到事件以后是如何处理的。
View的事件处理
对于这里描述的View,它是ViewGroup的父类,并不包含任何的子元素。这也就意味着View无法再次向下对事件进行分发操作,因此在View中并不存在onInterceptTouchEvent方法,也不会对事件做出拦截操作。它所做的事情就是对所接收的事件进行处理。下面就开看一下View如何对事件进行处理的。
既然ViewGroup将事件交由View的dispatchTouchEvent方。那么首先在这里就来看一下dispatchTouchEvent里面做了什么事情。
public boolean dispatchTouchEvent(MotionEvent event) {
......
if (onFilterTouchEventForSecurity(event)) {
//noinspection SimplifiableIfStatement
ListenerInfo li = mListenerInfo;
if (li != null && li.mOnTouchListener != null
&& (mViewFlags & ENABLED_MASK) == ENABLED
&& li.mOnTouchListener.onTouch(this, event)) {
result = true;
}
if (!result && onTouchEvent(event)) {
result = true;
}
}
......
return result;
}
在View的dispatchTouchEvent方法中对事件处理的核心部分体现在上述代码中。onFilterTouchEventForSecurity方法表示当前接收事件的view是否处于被遮盖状态,View处于被遮盖状态表示当前View不位于顶部,该view被其它View所覆盖。如果当前View被遮盖,那么该View不会对事件进行处理。
public interface OnTouchListener {
boolean onTouch(View v, MotionEvent event);
}
public void setOnTouchListener(OnTouchListener l) {
getListenerInfo().mOnTouchListener = l;
}
ListenerInfo getListenerInfo() {
if (mListenerInfo != null) {
return mListenerInfo;
}
mListenerInfo = new ListenerInfo();
return mListenerInfo;
}
在结合上述一段代码可以看到,通过setOnTouchListener方法设置OnTouchListener以后,若是当前View处于可用状态,那么条件li != null && li.mOnTouchListener !=null && (mViewFlags & ENABLED_MASK) == ENABLED必然为true。这时候程序便会回调OnTouchListener中的onTouch方法,若是在onTouch方法中返回true,便不会在执行View的onTouchEvent方法。从这里我们能够看到,一旦设置了OnTouchListener,那么OnTouchListener的优先级要高于onTouchEvent。
有一点需要注意,在程序中设置了OnTouchListener以后,对于OnTouchListener中的onTouch的返回值并不代表View中的dispatchTouchEvent方法所返回的值。在onTouch方法返回true的时候,表示事件成功被当前View所消耗,这时候result被置为true并且不再执行onTouchEvent,所以dispatchTouchEvent也就返回true。可是一旦在onTouch方法中返回false。这时候便会调用onTouchEvent方法,如果事件被onTouchEvent成功处理,并返回true,result依然会被置为true,dispatchTouchEvent自然而然的也就返回true。
下面在进入View的onTouchEvent方法一探究竟。对于onTouchEvent方法里的内容比较多,在这里分段查看。
if ((viewFlags & ENABLED_MASK) == DISABLED) {
if (action == MotionEvent.ACTION_UP && (mPrivateFlags & PFLAG_PRESSED) != 0) {
setPressed(false);
}
// A disabled view that is clickable still consumes the touch
// events, it just doesn't respond to them.
return (((viewFlags & CLICKABLE) == CLICKABLE
|| (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)
|| (viewFlags & CONTEXT_CLICKABLE) == CONTEXT_CLICKABLE);
}
在这里可以看出对于不可用的View,如果他们的一些点击事件可用的话,依然能够成功的消费事件,只是它不会为该事件做出响应。对于View的这些点击之间默认为不可用,但是对于不同的的View他们的默认值不一样。例如在ImageView中点击事件依然为不可用,但是在Button中点击事件为可用。当然如果手动为它们设置监听事件,那么这些监听事件都将会自动被设为可用状态。从如下源码中可以看出。
public void setOnClickListener(@Nullable OnClickListener l) {
if (!isClickable()) {
setClickable(true);
}
getListenerInfo().mOnClickListener = l;
}
public void setOnLongClickListener(@Nullable OnLongClickListener l) {
if (!isLongClickable()) {
setLongClickable(true);
}
getListenerInfo().mOnLongClickListener = l;
}
public void setOnContextClickListener(@Nullable OnContextClickListener l) {
if (!isContextClickable()) {
setContextClickable(true);
}
getListenerInfo().mOnContextClickListener = l;
}
下面接着看OnTouchEvent里面代码。
if (mTouchDelegate != null) {
if (mTouchDelegate.onTouchEvent(event)) {
return true;
}
}
这里首先判断是否对事件设置了代理,如果对事件设置了代理,便会执行TouchDelegate的onTouchEvent方法。mTouchDelegate默认值为null,可以通过View的setTouchDelegate方法来设置代理。对于TouchDelegate在后续的文章中在进行详细分析,在这里就不在过多描述。
最后看一下View是如何处理事件的,对于接收的事件整个处理过程比较复杂,在这里就从宏观上来整体看一下它的处理机制。
if (((viewFlags & CLICKABLE) == CLICKABLE ||
(viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE) ||
(viewFlags & CONTEXT_CLICKABLE) == CONTEXT_CLICKABLE) {
switch (action) {
case MotionEvent.ACTION_UP:
......
if (!mHasPerformedLongPress && !mIgnoreNextUpEvent) {
// This is a tap, so remove the longpress check
removeLongPressCallback();
// Only perform take click actions if we were in the pressed state
if (!focusTaken) {
// Use a Runnable and post this rather than calling
// performClick directly. This lets other visual state
// of the view update before click actions start.
if (mPerformClick == null) {
mPerformClick = new PerformClick();
}
if (!post(mPerformClick)) {
performClick();
}
}
}
......
break;
......
}
return true;
}
如果View的点击事件处于可用状态的话,便会对于这些事件进行处理,并且返回true。当一个事件序列完成以后调用performClick方法,下面看下这个performClick方法。
public boolean performClick() {
final boolean result;
final ListenerInfo li = mListenerInfo;
if (li != null && li.mOnClickListener != null) {
playSoundEffect(SoundEffectConstants.CLICK);
li.mOnClickListener.onClick(this);
result = true;
} else {
result = false;
}
sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);
return result;
}
从上面代码中可以看出,如果我们设置了OnClickListener,便会调用它的onClick方法。从这一路下来我们可以看出对于View的onClick事件,在最后才会被调用,可见onClick的优先级是最低的。
总结
在这里对View的事件处理做一下总结。在ViewGroup将事件分发到View以后。在View里面通过OnTouchListener的onTouch和View中的onTouchEvent这两个方法对事件进行处理。对于onTouch方法是View提供给用户的,方便用户自己处理触摸事件,而onTouchEvent是Android系统自己实现的接口。若是用户设置了OnTouchListener,Android系统会首先调用OnTouchListener的onTouch方法。若是在onTouch方法中返回true,就不在执行View的onTouchEvent方法。只有在onTouch中返回了false才会执行onTouchEvent。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Android事件分发机制
# View事件处理
# Android事件分发
# Android开发事件处理的代码如何写手摸手教程
# Android事件分发之View事件处理关键及示例分析
# Android XRecyclerView最简单的item点击事件处理
# 详解Android的两种事件处理机制
# Android 中 EventBus 的使用之多线程事件处理
# Android开发多手指触控事件处理
# 在这里
# 便会
# 可以看出
# 看一下
# 他们的
# 不可用
# 也就
# 才会
# 这时候
# 会对
# 就来
# 如何处理
# 这一
# 也不
# 将会
# 在这
# 设为
# 会在
# 它是
# 这也
相关文章:
如何用PHP快速搭建CMS系统?
如何在云主机上快速搭建网站?
如何在阿里云服务器自主搭建网站?
微信小程序 input输入框控件详解及实例(多种示例)
清除minerd进程的简单方法
长沙企业网站制作哪家好,长沙水业集团官方网站?
网站制作的方法有哪些,如何将自己制作的网站发布到网上?
整蛊网站制作软件,手机不停的收到各种网站的验证码短信,是手机病毒还是人为恶搞?有这种手机病毒吗?
如何通过IIS搭建网站并配置访问权限?
建站之星如何快速解决建站难题?
制作网站的软件下载免费,今日头条开宝箱老是需要下载怎么回事?
云南网站制作公司有哪些,云南最好的招聘网站是哪个?
javascript基本数据类型及类型检测常用方法小结
如何选择长沙网站建站模板?H5响应式与品牌定制哪个更优?
php能控制zigbee模块吗_php通过串口与cc2530 zigbee通信【介绍】
广州营销型建站服务商推荐:技术优势与SEO优化解析
股票网站制作软件,网上股票怎么开户?
西安专业网站制作公司有哪些,陕西省建行官方网站?
如何在Mac上搭建Golang开发环境_使用Homebrew安装和管理Go版本
利用JavaScript实现拖拽改变元素大小
,怎么用自己头像做动态表情包?
高防服务器租用首荐平台,企业级优惠套餐快速部署
专业的网站制作设计是什么,如何制作一个企业网站,建设网站的基本步骤有哪些?
济南专业网站制作公司,济南信息工程学校怎么样?
南宁网站建设制作定制,南宁网站建设可以定制吗?
XML的“混合内容”是什么 怎么用DTD或XSD定义
可靠的网站设计制作软件,做网站设计需要什么样的电脑配置?
如何选择PHP开源工具快速搭建网站?
如何在橙子建站上传落地页?操作指南详解
如何通过虚拟主机空间快速建站?
如何在阿里云购买域名并搭建网站?
保定网站制作方案定制,保定招聘的渠道有哪些?找工作的人一般都去哪里看招聘信息?
浅谈Javascript中的Label语句
广州顶尖建站服务:企业官网建设与SEO优化一体化方案
如何用狗爹虚拟主机快速搭建网站?
网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?
如何通过智能用户系统一键生成高效建站方案?
枣阳网站制作,阳新火车站打的到仙岛湖多少钱?
上海网站制作网站建设公司,建筑电工证网上查询系统入口?
建站主机数据库如何配置才能提升网站性能?
建站之星代理费用多少?最新价格详情介绍
黑客如何通过漏洞一步步攻陷网站服务器?
如何彻底删除建站之星生成的Banner?
建站之星备案是否影响网站上线时间?
全景视频制作网站有哪些,全景图怎么做成网页?
家庭建站与云服务器建站,如何选择更优?
建站之星多图banner生成与模板自定义指南
行程制作网站有哪些,第三方机票电子行程单怎么开?
在线流程图制作网站手机版,谁能推荐几个好的CG原画资源网站么?
如何正确选择百度移动适配建站域名?
*请认真填写需求信息,我们会在24小时内与您取得联系。