View的事件分发机制

1.前言

View的事件分发机制是面试的要点,也是必须要吃透的基础知识。虽然平时用到的地方不是那么频繁,但是一旦要用,如果这个不够扎实,就会卡手。就独立一篇来整理,日常开发中有发现有要注意的事情也在此补充归纳。
  与View做交互操作会产生事件,事件起点是谁,系统是如何将一个事件从起点传到目标View的。带着疑问去探索。

2.点击事件的传递规则

起始与流通:

从Android应用层看,事件起于Activity,Activity也有dispatchTouchEvent,onTouchEvent方法。
  Activity会传递给Window,Window会传给顶层View,顶层View分发给子View,一级级向下分发,如果某层不消费,则返给父层处理。
Activity -》 Window ->Decor View -> ContentView
  形象地来看,就如上级分派任务给下级。

三个核心方法:

  1. public boolean dispatchTouchEvent(MotionEvent ev)
    View都有。事件能传给当前View,则一定会调用.在这个方法会根据不同条件去调用onInterceptTouchEvent和onTouchEvent.返回值表示当前view是否消费了事件。
    true消费了;false没消费,事件交给父View处理.所有的父View都不处理,则传回Activity,在Activity中消亡。
  2. public boolean onInterceptTouchEvent(MotionEvent ev)
    仅ViewGroup有。返回值表示是否拦截事件。默认false不拦截,拦截则交给onTouch处理.不拦截交给子View处理(调用子View的dispatchTouchEvent())。
  3. public boolean onTouchEvent(MotionEvent ev)
    View都有。返回值表示对事件的处理。true表示处理了,false表示不处理。

改变这三个方法的返回值可以改变事件分发传递的过程,需要注意这样做默认的父类处理逻辑就不会执行了。

onTouchListener与方法优先级:

View可以设置onTouchListener,这个监听能干扰View的事件分发过程。它会先于onTounchEvent执行,也就是它的优先级高。
  优先级onTouchListener > onTouchEvent > onClickListener(具体View交互回调)
  onTouchListener中方法onTouch默认返回false,onTouchEvent会调用;true,不会调用OnTouchEvent,认为是当前View消费事件,即dispatchTouchEvent 返回true。
  无论onTouch怎样,dispatchTouchEvent最终都会调用,可以认为dispatchTouchEvent的优先级要高于onTouchListener。
View的dispatchTouchEvent()源码直观地体现了这一点.

事件与事件序列:

一个事件序列指手指从接触屏幕一刻起到离开屏幕这个过程产生的一系列事件。事件序列完整才会有相应的View交互回调方法执行。有ACTION_DOWN起。。。ACTION_UP止。
  正常情况下,一个事件序列交由一个View处理。一些特殊的手段可以让两个View都响应,比如在一个View的onTouchEvent中强行传递给其他View。

ViewGroup拦截ACTION_DOWN,ACTION_Down会传给自身的onTouchEvent.事件序列中之后事件不会再向下分发,传给自身的onTouchEvent.
ViewGroup拦截除ACTION_DOWN之外的事件,会产生一个ACTION_CANCEL的事件分发给子View.事件序列中之后的事件传给自身的onTouchEvent.此时mFirstTouchTarget为null,dispatchTouchEvent()不会调用onIntercepte()判断是否拦截,直接将intercepted赋值true.

View如果不消耗ACTION_DOWN事件(在onTounch或dispatchTouchEvent返回false),ACTION_DOWN事件和事件序列中之后的事件都交给父View处理.
View如果不消耗除ACTION_DOWN事件之外的事件,当前事件交给父View处理,之后还是会继续接收到事件序列中之后的事件.

子View用requestDisallowInterceptTouchEvent设置标记量mGroupFlags是否允许父View拦截事件,默认值是允许.每当有ACTION_DOWN事件来时会重置此标记量允许.所以面对ACTION_DOWN事件时,ViewGroup总会调用自己的InterceptTouchEvent方法来询问自己是否要拦截事件。当此标记量设置成不允许的时候,除ACTION_DOWN事件,dispatchTouchEvent()不会调用onIntercepte()判断是否拦截,直接将intercepted赋值false.

longClickable与clickable对View事件消耗的影响

View的onTouchEvent默认都会消耗事件,除非是不可点击的,即longClickable与clickable都为false.
  View的enable不影响事件分发.点击事件不会响应.
源码分析可以参考:
http://blog.csdn.net/weixin_37077539/article/details/54895485

可点击这个属性对View的事件消费有影响,在处理View的事件分发逻辑,记得检查View是否可点击.

dispatchTouchEvent()源码分析

原理在源码体现.
传送门:http://www.jianshu.com/p/93a060053cbc

3.滑动冲突实训

场景一:外部滑动方向与内部不一致。(外横内竖,viewpage已做外部拦截)
A:外部拦截法(常用)
重写父容器的onInterceptTouchEvent().当事件为ACTION_MOVE的时候。根据条件判断外部是否要拦截。ACTION_DOWN只在外层滑动将结束,优化滑动体验的时候才加上.

    boolean intercepted = false;
        int x = (int) ev.getX();
        int y = (int) ev.getY();
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                intercepted = false;
                if (!mScroller.isFinished()) {
                    mScroller.abortAnimation();
                    intercepted = true;
                }
                break;
            case MotionEvent.ACTION_MOVE:
                int deltaX = x - mLastXIntercept;
                int deltaY = y - mLastYIntercept;
                intercepted = Math.abs(deltaX) > Math.abs(deltaY);
                break;
            case MotionEvent.ACTION_UP:
               //给子View响应点击事件
                intercepted = false;
                break;
        }
        mLastX = x;
        mLastY = y;
        mLastXIntercept = x;
        mLastYIntercept = y;
        return intercepted;

B:内部拦截法
内部拦截要重写父View的onInterceptTouchEvent与子View的dispatchTouchEvent().利用的是子View可以用getParent.requestDisallowInterceptTouchEvent()方法改变父View的拦截标记位mGroupFlags,达到事件按照业务规则给自己或者父View处理的目的.
使用这个方法需要注意
1.父View不能将ACTION_DOWN拦截掉,ACTION_DOWN一旦被拦截,子View得不到事件序列后续的事件.
2.较外部拦截的效果需要处理内部滑动后不再将事件给父View.

 //内部拦截父View需要的代码
        mLastX = (int) ev.getRawX();
        mLastY = (int) ev.getRawY();
        if (MotionEvent.ACTION_DOWN == ev.getAction()) {
            if (!mScroller.isFinished()) {
                mScroller.abortAnimation();
                return true;
            }
            return false;
        } else {
            return true;
        }
    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        float x = ev.getRawX();
        float y = ev.getRawY();
        if (MotionEvent.ACTION_DOWN == ev.getAction()) {
            getParent().requestDisallowInterceptTouchEvent(true);
            slop = false;
        }
        if (MotionEvent.ACTION_MOVE == ev.getAction()) {
            if (Math.abs(mLastY - y) < Math.abs(mLastX - x)) {
                if (!slop) getParent().requestDisallowInterceptTouchEvent(false);
            } else {
                int touchSlop = ViewConfiguration.get(getContext()).getScaledTouchSlop();
                if (Math.abs(y - mLastY) > touchSlop) {
                    //已经开始了竖向滑动
                    slop = true;
                }
            }
        }
        mLastX = x;
        mLastY = y;
        return super.dispatchTouchEvent(ev);
    }

场景二:外部滑动方向与内部一致。
同场景一处理方式,区别在判断条件由业务定,看业务什么时候需要交给外层,什么时候需要交给内层。

场景三:场景一场景二交替同时出现
剥茧法层层解决.
可以加一些回弹或过渡效果使滑动冲突处理的过程显得更加圆滑。

后记

分发机制比较灵活,比较细腻,花我老长时间去分析理解巩固,源码风格不怎么利于阅读...233.有些情况套用公式是不行的,要多动手结合真实项目代码思考练习.
如果碰到不懂的地方,翻源码思考是一个比较好的选择.
Action_Cancle事件怎么产生与如何影响分发的,待做.
接下来会想对项目中RecyclerView左滑的实例进行思考探究.在这之前也需要有一定的View的绘制机制基础.

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

推荐阅读更多精彩内容