带你轻松理解Android事件分发机制


引入:在Android设备中,触摸事件主要包括点按(单击和双击)、长按、拖拽、滑动等,另外还包括单指操作和多指操作等。Android把这些操作抽象成MotionEvent这一概念。
常用:常用的是手指按下(ACTION_DOWN)、滑动(ACTION_MOVE)、抬起(ACTION_UP)。我们平时最简单的操作是包括一个ACTION_DOWN事件,多个ACTION_MOVE,一个ACTION_UP组成。

通常事件的传递流程:

Activity的dispatchTouchEvent开始--- PhoneWindow---DecorView(顶级view)---交由viewgroup--view

ViewGroup

在View上触发事件,最先捕获到事件的为View所在的ViewGroup,然后才会到View自身,假如此时我们在一个linearlayout上有一个button,现在点击button,会先找到linearlayout。
流程:MyLinearLayout的dispatchTouchEvent -> MyLinearLayout的onInterceptTouchEvent -> MyButton的dispatchTouchEvent ->Mybutton的onTouchEvent
  • 下面放一段viewgroup的dispatchTouchEvent的源码
@Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        if (!onFilterTouchEventForSecurity(ev)) {
            return false;
        }

        final int action = ev.getAction();
        final float xf = ev.getX();
        final float yf = ev.getY();
        final float scrolledXFloat = xf + mScrollX;
        final float scrolledYFloat = yf + mScrollY;
        final Rect frame = mTempRect;

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

        if (action == MotionEvent.ACTION_DOWN) {
            if (mMotionTarget != null) {
                // this is weird, we got a pen down, but we thought it was
                // already down!
                // XXX: We should probably send an ACTION_UP to the current
                // target.
                mMotionTarget = null;
            }
            // If we're disallowing intercept or if we're allowing and we didn't
            // intercept
            if (disallowIntercept || !onInterceptTouchEvent(ev)) {
                // reset this event's action (just to protect ourselves)
                ev.setAction(MotionEvent.ACTION_DOWN);
                // We know we want to dispatch the event down, find a child
                // who can handle it, start with the front-most child.
                final int scrolledXInt = (int) scrolledXFloat;
                final int scrolledYInt = (int) scrolledYFloat;
                final View[] children = mChildren;
                final int count = mChildrenCount;

                for (int i = count - 1; i >= 0; i--) {
                    final View child = children[i];
                    if ((child.mViewFlags & VISIBILITY_MASK) == VISIBLE
                            || child.getAnimation() != null) {
                        child.getHitRect(frame);
                        if (frame.contains(scrolledXInt, scrolledYInt)) {
                            // offset the event to the view's coordinate system
                            final float xc = scrolledXFloat - child.mLeft;
                            final float yc = scrolledYFloat - child.mTop;
                            ev.setLocation(xc, yc);
                            child.mPrivateFlags &= ~CANCEL_NEXT_UP_EVENT;
                            if (child.dispatchTouchEvent(ev))  {
                                // Event handled, we have a target now.
                                mMotionTarget = child;
                                return true;
                            }
                            // The event didn't get handled, try the next view.
                            // Don't reset the event's location, it's not
                            // necessary here.
                        }
                    }
                }
            }
        }                                                                                                                                                  ....//other code omitted
  • 首先在linearlayout中的dispatchTouchEvent方法中:判断是否拦截:if (!disallowIntercept && onInterceptTouchEvent(ev)) 当前允许拦截且拦截了 只有两个条件都满足才返回true,表示拦截,不会向下进行。默认是不拦截的。

  • onInterceptTouchEvent这个方法是viewgroup特有的。true拦截,将不会分发给子view处理;false不拦截,可继续向下分发。默认是不拦截。若想拦截,如下:
public boolean onInterceptTouchEvent(MotionEvent ev)
    {
        int action = ev.getAction();
        switch (action)
        {
        case MotionEvent.ACTION_DOWN:
            //如果你觉得需要拦截
            return true ; 
        case MotionEvent.ACTION_MOVE:
            //如果你觉得需要拦截
            return true ; 
        case MotionEvent.ACTION_UP:
            //如果你觉得需要拦截
            return true ; 
        }
        
        return false;
    }

如果你在DOWN retrun true ,则DOWN,MOVE,UP子View都不会捕获事件;如果你在MOVE return true , 则子View在MOVE和UP都不会捕获事件。
⚠️ :如果linearlayout已经进行拦截,但是子view还是想能够响应,这时我们即可用到android提供我们的方法, getParent().requestDisallowInterceptTouchEvent(true);去表示是否允许拦截。这样即使ViewGroup在MOVE的时候return true,子View依然可以捕获到MOVE以及UP事件。不加这一句子view就将被拦截,button将响应不到事件。这个和我们第一点就上了,判断是否拦截:if (!disallowIntercept && onInterceptTouchEvent(ev)) 当前允许拦截且拦截了 只有两个条件都满足才表示拦截。
  • 若dispatchTouchEvent返回false,表示不拦截,此时父viewgroup则开始查找当前x,y是否在某个子View的区域内,如果在,则把事件分发下去。
  • 若没有找到子view,就由viewgroup担任view,直接super.dispatchTouchEvent(ev);
  • 从这段代码看,只有子View.dispatchTouchEvent(ev)返回的为true;才算真正找到去响应事件的真正视图。此时mMotionTarget才不为空, 此时父view的dispatchTouchEvent(ev)将返回true,表示已经消费,不用再继续向下进行找view了。重点理解!!
                     if (child.dispatchTouchEvent(ev))  {
                                // Event handled, we have a target now.
                                mMotionTarget = child;
                                return true;
                            }

View

传递到view,有三个重要的方法需要理解与掌握,这三个方法也有顺序:

View.dispatchEvent->
View.setOnTouchListener(ontouch)->
View.onTouchEvent

  • 在dispatchTouchEvent中mOnTouchListener!=null 并且mOnTouchListener.onTouch(this, event)返回true 。返回true,消费,不分发,onTouchEvent不会被执行。否则执行onTouchEvent。
  • 说明:如果在ontouch中return true (表示消费,不继续分发),ontouchevent将接收不到touch事件、相应的点击事件也不会生效。(因为onclick事件是在ontouchevent中调用)

  • dispatchTouchEvent:用于事件的分发,所有的事件都要通过此方法进行分发,决定是自己对事件进行消费还是交由子View处理. false表示继续分发、传递给子view。true表示不继续分发,消费。
  • onInterceptEvent:是ViewGroup中独有的方法,若返回true表示拦截当前事件,交由自己的onTouchEvent()进行处理,返回false表示不拦截。
  • onTouchEvent: 主要用于事件的处理,返回true表示消费当前事件

理解完文章再理解一下这张图吧~结合图和文章要点,如果你懂了,说明你已经get了!

事件分发流程
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,052评论 6 513
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,267评论 3 397
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 167,518评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,457评论 1 296
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,474评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,143评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,728评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,650评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,184评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,269评论 3 339
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,415评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,083评论 5 348
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,775评论 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,257评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,389评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,777评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,419评论 2 359

推荐阅读更多精彩内容