android事件拦截和分发机制详解

上图,Android整个事件分发机制:


一:事件分发流程

Activity->viewGroup->子view

ViewGroup 及其子类对与 Touch 事件相关的三个方法均能响应,而 Activity 对 onInterceptTouchEvent(MotionEvent ev) 也就是事件拦截不进行响应。另外需要注意的是 View 对 dispatchTouchEvent(MotionEvent ev) 和onInterceptTouchEvent(MotionEvent ev) 的响应的前提是可以向该 View 中添加子 View,如果当前的 View 已经是一个最小的子View(比如 Imageview,Button),那么就无法向这个最小 View 中添加子 View,也就无法向子 View 进行事件的拦截,所以它没有onInterceptTouchEvent(MotionEvent ev)。

如图执行顺序是:Activity-1->2->3->4->5->6->7
需要注意的是:activity没有onInterceptTouchEvent方法,子view中也没有onInterceptTouchEvent方法。

二:与事件分发的三个重要回调方法

1.   dispatchTouchEvent

return true,事件会分发给当前 View并由 dispatchTouchEvent方法进行消费,同时事件会停止向下传递;

 return false,事件分发分为两种情况:

如果当前 View 获取的事件直接来自Activity,则会将事件返回给Activity onTouchEvent 进行消费;如果当前View 获取的事件来自外层父控件,则会将事件返回给父View   onTouchEvent进行消费。如果返回系统默认的 super.dispatchTouchEvent(ev),事件会自动的分发给当前ViewonInterceptTouchEvent 方法。

2 .  onInterceptTouchEvent

ViewdispatchTouchEvent(MotionEvent ev)方法返回系统默认的 super.dispatchTouchEvent(ev)情况下,事件会自动的分发给当前 ViewonInterceptTouchEvent方法。逻辑返回:

true,则表示将事件进行拦截,并将拦截到的事件交由当前 ViewonTouchEvent进行处理

false,则表示将事件放行,当前 View上的事件会被传递到子View上,再由子 ViewdispatchTouchEvent来开始这个事件的分发;返回 super.onInterceptTouchEvent(ev),事件默认会被拦截,并将拦截到的事件交由当前View onTouchEvent 进行处理

3. ontouchEvent:

dispatchTouchEvent返回super.dispatchTouchEvent(ev)并且onInterceptTouchEvent返回true或返回super.onInterceptTouchEvent(ev)的情况下onTouchEvent会被调用。onTouchEvent 的事件响应逻辑如下:

如果事件传递到当前 View onTouchEvent 方法,而该方法返回了 false,那么这个事件会从当View向上传递并且都是由上层ViewonTouchEvent来接收,如果传递到上面的onTouchEvent 也返回false,这个事件就会消失,而且接收不到下一次事件。

如果返回了 true 则会接收并消费该事件。

如果返回 super.onTouchEvent(ev) 默认处理事件的逻辑和返回 false 时相同。

三:dispatchTouchEvent,ontouchEvent,ontouchListener,onclickListener之间的区别和联系

如果一个给一个button同时设置了click和toutch事件,OntouchListener需要一个返回值,如果返回了true,表示把事件消费掉了,onclick事件就不会执行;如果返回false,toutch和click事件能够同时响应。因为:在dispatchTouchEvent方法中
public boolean dispatchTouchEvent(MotionEvent event) {  
if (li != null && li.mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED  
                 && li.mOnTouchListener.onTouch(this, event)) {  
                return true;  
            }
        return false;  
    }
而在ontouchEvent方法里面调用performclick()方法,onclick在performClick中执行,所以ontouch的优先级比onclick优先级高。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值