安卓事件分发?超简单~

作者: space0o0 | 来源:发表于2018-03-24 21:41 被阅读108次

    平时写代码,以业务为主,作为一个菜逼,总是觉得把工作做完就行,也就没那么关注原理。随着工作年龄的增加,对各个控件的原理也就是模模糊糊的,讲不清,道不明。某天突然脑抽,想写一系列进阶的文章,巩固下基础,补全那乱七八糟的理解。

    今天来探讨一下view的分发机制。

    平时写xml,对控件的使用就是一个layout包个layout,一个节点一个节点的下来。可系统怎么知道我点的是这个控件,而不是包着他的那个父控件呢?

    咱们先来个生动形象的爷爷爸爸儿子控件。


    全家福.png

    这其中代表着最普遍的嵌套关系
    爷爷:rootlayout(根布局)
    爸爸:viewgroup(内部的layout等)
    儿子:view(如button,textview等)

    咱们非常清楚,这里爷爷最年长,所以爸爸要听爷爷的话。
    那么儿子就要听爸爸的话。

    所以我们的事件分发顺序就是:
    rootlayout(“爷爷”) -> viewgroup(“爸爸”) -> view(“儿子”)

    每个成员都有dispatchTouchEvent(MotionEvent ev)方法,用来分发事件。

    “儿子”最小,事件到了他这里,他不能拦截,要么消费掉,要么再回传,因为“儿子”没地方分发事件,也就没有可拦截的地方。

    方法 rootLayout viewGroup view
    dispatchTouchEvent(分发)
    onInterceptTouchEvent(拦截) 没有
    onTouchEvent(消费)
    消费路径.png
    • 我们把event看做是“糖果”

    爷爷得到了一个糖果,进入dispatchTouchEvent(),调用了onInterceptTouchEvent(),如果合口味,那就返回true,拦截了该事件,交给onTouchEvent()处理;不拦截,就传递给下一个viewgroup

    //    rootlayout
    
    @Override
        public boolean dispatchTouchEvent(MotionEvent ev) {
            if (ev.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "rootview dispatchTouchEvent: 我得到了一个糖果");
            }
            return super.dispatchTouchEvent(ev);
        }
    
        @Override
        public boolean onInterceptTouchEvent(MotionEvent ev) {
            if (ev.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "rootview onInterceptTouchEvent:true 我要自己吃这个糖果");
                return true;
            }
            return super.onInterceptTouchEvent(ev);
        }
    
        @Override
        public boolean onTouchEvent(MotionEvent event) {
            if (event.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "rootview onTouchEvent: 爷爷自己吃了糖果");
            }
            return super.onTouchEvent(event);
        }
    
     D////爷爷///: rootview dispatchTouchEvent: 我得到了一个糖果
     D////爷爷///: rootview onInterceptTouchEvent:true 我要自己吃这个糖果
     D////爷爷///: rootview onTouchEvent: 爷爷自己吃了糖果
    

    现在糖果不合“爷爷”口味,太硬了,磕牙~
    事件传递给“爸爸”。

    // viewgroup
    @Override
        public boolean dispatchTouchEvent(MotionEvent ev) {
            if (ev.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "viewGroup dispatchTouchEvent: 爸爸从爷爷那得到一颗糖~");
            }
            return super.dispatchTouchEvent(ev);
        }
    
        @Override
        public boolean onInterceptTouchEvent(MotionEvent ev) {
            if (ev.getAction()==MotionEvent.ACTION_DOWN){
                Log.d(TAG, "viewGroup onInterceptTouchEvent:我也不要吃 ");
            }
            return super.onInterceptTouchEvent(ev);
        }
    
        @Override
        public boolean onTouchEvent(MotionEvent event) {
            if (event.getAction()==MotionEvent.ACTION_DOWN){
                Log.d(TAG, "viewGroup onTouchEvent: 我消费~");
            }
            return super.onTouchEvent(event);
        }
    
    // view
        @Override
        public boolean dispatchTouchEvent(MotionEvent event) {
            if (event.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖 ");
            }
            return super.dispatchTouchEvent(event);
        }
    
        @Override
        public boolean onTouchEvent(MotionEvent event) {
            if (event.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "View1 onTouchEvent: true 儿子吃了糖果");
                return true;
            }
            return super.onTouchEvent(event);
        }
    
    D////爷爷///: rootview dispatchTouchEvent: 我得到了一个糖果
    D////爷爷///: rootview onInterceptTouchEvent:false 我不要吃这个糖果
    D////爸爸///: viewGroup dispatchTouchEvent: 爸爸从爷爷那得到一颗糖~
    D////爸爸///: viewGroup onInterceptTouchEvent:我也不要吃 
    D////儿子///: View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖 
    D////儿子///: View1 onTouchEvent: true 儿子吃了糖果
    

    如果“儿子”也不要吃糖果,onTouchEvent返回false,回传给上一层的onTouchEvent,根据返回的true或者false来决定是否要消费。

    //  view
    @Override
        public boolean dispatchTouchEvent(MotionEvent event) {
            if (event.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖 ");
            }
            return super.dispatchTouchEvent(event);
        }
    
        @Override
        public boolean onTouchEvent(MotionEvent event) {
            if (event.getAction() == MotionEvent.ACTION_DOWN) {
                Log.d(TAG, "View1 onTouchEvent: false 我不要吃");
                return false;
            }
            return super.onTouchEvent(event);
        }
    
    D////爷爷///: rootview dispatchTouchEvent: 我得到了一个糖果
    D////爷爷///: rootview onInterceptTouchEvent:false 我不要吃这个糖果
    D////爸爸///: viewGroup dispatchTouchEvent: 爸爸从爷爷那得到一颗糖~
    D////爸爸///: viewGroup onInterceptTouchEvent:我也不要吃 
    D////儿子///: View1 dispatchTouchEvent: 儿子从爸爸那得到一颗糖 
    D////儿子///: View1 onTouchEvent: false 我不要吃
    D////爸爸///: viewGroup onTouchEvent: true 儿子不吃,那我吃
    

    至此,事件分发的大致流程在爷孙三谦让的分糖果过程中完全体现。如有错误处与改进处,请给我留言,谢谢~

    相关文章

      网友评论

      本文标题:安卓事件分发?超简单~

      本文链接:https://www.haomeiwen.com/subject/xsjfcftx.html