Android事件分发机制以及滑动冲突处理

2024-08-26 06:48

本文主要是介绍Android事件分发机制以及滑动冲突处理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

转载请注明出处:http://blog.csdn.net/u013038616/article/details/50733811

方便日后的查看与交流,将学习与实践总结如下。


一、Android事件传递分析

1、ViewGroup中事件分发机制相关的方法
a、dispatchTouchEvent  事件分发器
b、onInterceptTouchEvent  处理是否拦截事件
c、onTouchEvent 处理对应的事件

ViewGroup中他们的代码关系如下伪代码表示:

public boolean dispatchTouchEvent(MotionEvent ev){boolean consume = false;if(onInterceptTouchEvent(ev)){consume = onTouchEvent(ev);}else{consume = child.dispatchTouchEvent(ev);}return consume;}	

常用结论:
a、一个完整的MotionEvent由DOWN(1)->MOVE(n>=0)->UP(1)构成
b、一旦onInterceptTouchEvent拦截了某类事件,该onInterceptTouchEvent方法只会调用一次,后续该事件默认交给该View执行。例如某ViewGroup拦截了DOWN事件,后面的MOVE(n>=0)->UP(1)就默认由该ViewGroup处理,不会向下传递。
c、事件的传递方向:由父容器传向子View,即由外到内。
d、View中没有onInterceptTouchEvent()方法,View默认自己处理事件。
e、ViewGroup的onInterceptTouchEvent()方法默认返回false,即不拦截任何事件。


2、事件分发过程相关类和方法的源码
a、事件传递的大体过程
Activity->Window->View

b、Activity的事件分发器源代码

/*** Called to process touch screen events.  You can override this to* intercept all touch screen events before they are dispatched to the* window.  Be sure to call this implementation for touch screen events* that should be handled normally.** @param ev The touch screen event.** @return boolean Return true if this event was consumed.*/public boolean dispatchTouchEvent(MotionEvent ev) {if (ev.getAction() == MotionEvent.ACTION_DOWN) {onUserInteraction();}if (getWindow().superDispatchTouchEvent(ev)) {return true;}return onTouchEvent(ev);}
可以看出Activity先将事件交给Window的superDispatchTouchEvent(ev)处理,如果返回true则事件处理结束,否则事件将最终由Activity的onTouchEvent(ev)方法处理。

c、Window的superDispatchTouchEvent(ev)方法

/*** Used by custom windows, such as Dialog, to pass the touch screen event* further down the view hierarchy. Application developers should* not need to implement or call this.**/public abstract boolean superDispatchTouchEvent(MotionEvent event);
Window是一个抽象类,唯一实现在android.policy.PhoneWindow,Windows的类描述原文如下:

/*** Abstract base class for a top-level window look and behavior policy.  An* instance of this class should be used as the top-level view added to the* window manager. It provides standard UI policies such as a background, title* area, default key processing, etc.** <p>The only existing implementation of this abstract class is* android.policy.PhoneWindow, which you should instantiate when needing a* Window.  Eventually that class will be refactored and a factory method* added for creating Window instances without knowing about a particular* implementation.*/
在android.policy.PhoneWindow中superDispatchTouchEvent方法实现为:

public boolean superDispatchTouchEvent(MotionEvent event){return mDecor.superDispatchTouchEvent(event);}
该方法调用了顶级View(DecorView)的superDispatchTouchEvent(event)方法进行事件分发,顶级View继承自FrameLayout,通过setContentView设置的View为DecorView的子类。
DecorView的类定义如下:
private final class DecorView extends FrameLayout implements RootViewSurfaceView{//...}
因为DecorView也是ViewGroup所以事件分发的过程和ViewGroup的事件分发过程类似。


d、View中dispatchTouchEvent方法主要的分发逻辑

/*** Pass the touch screen motion event down to the target view, or this* view if it is the target.** @param event The motion event to be dispatched.* @return True if the event was handled by the view, false otherwise.*/public boolean dispatchTouchEvent(MotionEvent event) {//......boolean result = false;//......if (onFilterTouchEventForSecurity(event)) {//noinspection SimplifiableIfStatementListenerInfo 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不是ViewGroup不需要向其子View进行事件分发,所以View中没有onInterceptTouchEvent方法,事件将直接交给OnTouchListener的onTouch或ViewGroup的onTouchEvent方法处理,且OnTouchListener会屏蔽onTouchEvent方法。

e、ViewGroup中dispatchTouchEvent方法主要的分发逻辑
ViewGroup的分发代码比较复杂,暂不列出,有兴趣的话可以去看看源码。


3、一个完整的事件分发过程(该事件的起点区域必须有重叠的子View,父容器才会进行事件分发,否则
父类将自己处理该事件的全部过程)

a、事件先传递给Activity的public boolean dispatchTouchEvent(MotionEvent ev)
b、Activity先 将事件交给Window类的superDispatchTouchEvent(ev)处理
c、Window类再将事件交给顶级View(DecorView)进行事件传递处理
d、顶级View(DecorView)将事件传递给我们定义的布局文件中View(一般为ViewGroup)处理,如果处
理则将事件,即ViewGroup的onInterceptTouchEvent方法返回true,则事件有该ViewGroup处理,如果设
置了setOnTouchEventListener,那么将调用onTouch方法,否则调用其onTouchEvent方法。如果setOnClickListener
点击事件则点击事件被调用;如果不处理该事件,则该事件将向下传递给其子View,调用子View的
dispatchTouchEvent方法,依次处理直到事件被处理,如果最终未处理则进行e。
e、getWindow().superDispatchTouchEvent(ev)返回了false就说明没有View处理该事件,事件将最
终传递给Activity的onTouchEvent进行处理。


二、处理View的滑动冲突

1、滑动冲突的分类
滑动冲突的解决方法是根据上面事件传递的机制进行处理的,熟练掌握事件分发机制是处理滑动冲突的前提。然后就是根据不同的规则来重写View的onInterceptTouchEvent方法,来决定什么时候让父容器拦截滑动事件什么时候让子View拦截滑动事件。

滑动冲突大致可分以下两类:
1、两层滑动嵌套的情况:
a、外层与内层的滑动方向垂直
b、外层与内层的滑动方向平行
2、多层滑动嵌套的情况:可以转化为两层的情况进行处理

滑动冲突的处理方式可分以下两类:
a、外部拦截法:所有事件都要经过父容器,如果父容器需要此事件就进行拦截,不需要就不拦截。
b、内部拦截法:所有事件都传递给子View,如果子View需要此事件就直接消耗,如果不需要就交给
父容器处理。该方法与原有的分发顺序不一样,需要配合parent.requestDisallowInterceptTouchEvent()进行事件重新分发才能正常工作。


2、滑动冲突的实例解决方案
滑动冲突第一类中的第一种的解决方法如下:
如父容器需要左右滑动,而子View需要上下滑动(如ViewPager的情况)。这个解决方法很简单也很典型,直接可以通过用户的滑动来判断用户的意图。通过判断手指在屏幕上移动一小段距离,计算出x轴上的增量dx,y轴上的增量dy,比较dx与dy绝对值的大小,如果dx>dy说明用户想左右滑动;反之,是想上下滑动。这是最直接的一种判断方法,还可以根据滑动方向与水平方向的夹角判断,总之能正确区分用户的意图即可。

a、采用外部拦截法
然后根据上面的规则重写父容器的onInterceptTouchEvent方法,子View不用做任何求改。如下:

// 分别记录上次滑动的坐标private int mLastXIntercept = 0;private int mLastYIntercept = 0;@Overridepublic boolean onInterceptTouchEvent(MotionEvent event) {boolean intercepted = false;int x = (int) event.getX();int y = (int) event.getY();switch (event.getAction()) {case MotionEvent.ACTION_DOWN: {//这里将返回值设为false是为了让子View可以接受到事件//如果为true则父容器将处理整个事件,子View将接受不到事件intercepted = false;break;}case MotionEvent.ACTION_MOVE: {//根据x轴,y轴方向增量值判断(可以根据不同的规则进行修改)int deltaX = x - mLastXIntercept;int deltaY = y - mLastYIntercept;if (Math.abs(deltaX) > Math.abs(deltaY)) {intercepted = true;} else {intercepted = false;}break;}case MotionEvent.ACTION_UP: {intercepted = false;break;}default:break;}mLastXIntercept = x;mLastYIntercept = y;return intercepted;}

b、采用内部拦截法
该方法需要改变事件的分发顺序,所以需要重写子View的dispatchTouchEvent方法。根据上面的规则子View的dispatchTouchEvent方法改写如下:
注意:其中parent为该View需要拦截滑动事件的那个父容器的引用。

// 分别记录上次滑动的坐标private int mLastX = 0;private int mLastY = 0;@Overridepublic boolean dispatchTouchEvent(MotionEvent event) {int x = (int) event.getX();int y = (int) event.getY();switch (event.getAction()) {case MotionEvent.ACTION_DOWN: {//不允许父容器拦截该事件//parent为该View需要拦截滑动事件的那个父容器的引用parent.requestDisallowInterceptTouchEvent(true);break;}case MotionEvent.ACTION_MOVE: {//这里的逻辑可以根据需要进行修改//parent为该View需要拦截滑动事件的那个父容器的引用int deltaX = x - mLastX;int deltaY = y - mLastY;if (Math.abs(deltaX) > Math.abs(deltaY)) {//允许父容器拦截该事件parent.requestDisallowInterceptTouchEvent(false);}break;}case MotionEvent.ACTION_UP: {break;}default:break;}mLastX = x;mLastY = y;return super.dispatchTouchEvent(event);}

父容器也要做相应的修改:
父容器需要拦截处理ACTION_DOWN以外的所有事件。然后通过在子View中调用requestDisallowInterceptTouchEvent方法来控制父容器是否截断事件。

@Overridepublic boolean onInterceptTouchEvent(MotionEvent event) {int action = event.getAction();if (action == MotionEvent.ACTION_DOWN) {return false;} else {return true;}}
这里父容器如果拦截ACTION_DOWN事件,那么整个事件将交给父容器处理,子View将接受不到任何事件,从而不能进行内部拦截。

滑动冲突第一类中的第二种的解决方法和第一种的解决方法类似,虽然不能从用户点击屏幕的操作直接判断出用户的意图,但是可以根据不同的业务逻辑加以区分,只要能确定什么时候让父容器截获事件什么时候子View截获事件,就将上面方法的标注部分就行相应的修改即可处理此类冲突。


第二类处理起来比较复杂一些,需要将多层的冲突分解为第一类中单个的事件冲突,对单个的事件冲突再进项逻辑判断上的处理,就可将大化小,一层一层的处理冲突,即可解决此类滑动冲突。
















这篇关于Android事件分发机制以及滑动冲突处理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1107849

相关文章

如何使用celery进行异步处理和定时任务(django)

《如何使用celery进行异步处理和定时任务(django)》文章介绍了Celery的基本概念、安装方法、如何使用Celery进行异步任务处理以及如何设置定时任务,通过Celery,可以在Web应用中... 目录一、celery的作用二、安装celery三、使用celery 异步执行任务四、使用celery

SpringBoot操作spark处理hdfs文件的操作方法

《SpringBoot操作spark处理hdfs文件的操作方法》本文介绍了如何使用SpringBoot操作Spark处理HDFS文件,包括导入依赖、配置Spark信息、编写Controller和Ser... 目录SpringBoot操作spark处理hdfs文件1、导入依赖2、配置spark信息3、cont

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Spring使用@Retryable实现自动重试机制

《Spring使用@Retryable实现自动重试机制》在微服务架构中,服务之间的调用可能会因为一些暂时性的错误而失败,例如网络波动、数据库连接超时或第三方服务不可用等,在本文中,我们将介绍如何在Sp... 目录引言1. 什么是 @Retryable?2. 如何在 Spring 中使用 @Retryable

基于Redis有序集合实现滑动窗口限流的步骤

《基于Redis有序集合实现滑动窗口限流的步骤》滑动窗口算法是一种基于时间窗口的限流算法,通过动态地滑动窗口,可以动态调整限流的速率,Redis有序集合可以用来实现滑动窗口限流,本文介绍基于Redis... 滑动窗口算法是一种基于时间窗口的限流算法,它将时间划分为若干个固定大小的窗口,每个窗口内记录了该时间

MyBatis延迟加载的处理方案

《MyBatis延迟加载的处理方案》MyBatis支持延迟加载(LazyLoading),允许在需要数据时才从数据库加载,而不是在查询结果第一次返回时就立即加载所有数据,延迟加载的核心思想是,将关联对... 目录MyBATis如何处理延迟加载?延迟加载的原理1. 开启延迟加载2. 延迟加载的配置2.1 使用

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

Python中处理NaN值的技巧分享

《Python中处理NaN值的技巧分享》在数据科学和数据分析领域,NaN(NotaNumber)是一个常见的概念,它表示一个缺失或未定义的数值,在Python中,尤其是在使用pandas库处理数据时,... 目录NaN 值的来源和影响使用 pandas 的 isna()和 isnull()函数直接比较 Na

Python中的异步:async 和 await以及操作中的事件循环、回调和异常

《Python中的异步:async和await以及操作中的事件循环、回调和异常》在现代编程中,异步操作在处理I/O密集型任务时,可以显著提高程序的性能和响应速度,Python提供了asyn... 目录引言什么是异步操作?python 中的异步编程基础async 和 await 关键字asyncio 模块理论

详解Python中通用工具类与异常处理

《详解Python中通用工具类与异常处理》在Python开发中,编写可重用的工具类和通用的异常处理机制是提高代码质量和开发效率的关键,本文将介绍如何将特定的异常类改写为更通用的ValidationEx... 目录1. 通用异常类:ValidationException2. 通用工具类:Utils3. 示例文