Android事件分发机制

事件分发相对于ViewGroup来讲的,节点View不能分发事件。
1.dispatchTouchEvent(MotionEvent event)
2.onInterceptTouchEvent(MotionEvent event)
3.onTouchEvent(MotionEvent event)

事件——MotionEvent

public boolean dispatchTouchEvent(MotionEvent ev)
用于对事件进行分发。如果事件能被传递给当前View,那么该方法一定会被调用并且是首先被调用,返回值代表当前事件是否被消耗,
public boolean onInterceptTouchEvent(MotionEvent ev)
用于判断是否拦截某个事件,在上述方法的内部被调用。如果View拦截了某个事件,那么在同一事件序列中,此方法不会被再次调用。
public boolean onTouchEvent(MotionEvent ev)
用于对事件的具体处理,返回值表示是否消耗当前事件。如果onTouchEvent()方法未消耗当前事件,则在同一事件序列中,当前View无法再次接收到事件。

onInterceptTouchEvent(MotionEvent event)是ViewGroup的一个方法,系统向ViewGroup及其所有子视图传递事件之前进行一次拦截,不能包含子view的控件是没有这个方法的,如TextView就没有。

Event Down事件首先会传递到onInterceptTouchEvent(MotionEvent event)
Event Down
return false/true;
Move
return false/true;
Up
return false/true;

如果ViewGroup的onInterceptTouchEvent(Event event)接收到Down
事件处理完成之后
return true
那么MotionEvent后续的Move, Up等事件将不再分发,不再由onInterceptTouchEvent()分发下去了,而是和Down事件一起传递给ViewGroup
onTouchEvent(MotionEvent event)处理,注意,目标view将接收不到任何事件。
onTouchEvent(MotionEvent event)
Event Down
return false/true;
Move
return false/true;
Up
return false/true;

如果ViewGroup的onInterceptTouchEvent()在接收到
down事件
处理完成之后
return false
那么后续的move, up等事件将继续由onInterceptTouchEvent()分发,之后跟down事件一样传递给最终的目标view的onTouchEvent()处理。

这里需要说明onInterceptTouchEvent()在接收到Move事件完成后return true

如果最终需要处理事件的view的onTouchEvent()
Down
处理完事件
return false
那么该事件将被传递至其上一层次的view的onTouchEvent()处理。

如果最终需要处理事件的view的onTouchEvent()
Down
处理完事件
return true
那么后续事件将继续传递给该view的onTouchEvent()处理。

如果ViewGroup中onInterceptTouchEvent()拦截Down事件
return true,说明拦截了干事件由我来处理,不会向子View分发了。所以不会再由onInterceptTouchEvent()函数分发事件,也就不会再调用该函数,子View无法接受到任何的Touch事件,导致目标View不会收到Touch事件。全部由该ViewGroup中的onTouchEvent消耗掉。
如果onTouchEvent拦截
Down事件
return true,则该ViewGroup中结束全部事件,即Move 、UP结束,不会执行Move及Up事件。
如果该ViewGroup有父View,把Down Move Up三种事件返回给父类View处理,父View没有子View则无需调用onInterceptTouchEvent()分发函数,没有父类,则结束事件。

//拦截按下事件
    @Override
    public boolean onInterceptTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN: {
                Log.d(TAG, "MotionEvent.ACTION_DOWN");
                return true;
            }
    }
//所有Touch事件被该onTouchEvent消耗
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN: {
                Log.d(TAG, "MotionEvent.ACTION_DOWN");
                return true;
            }
            case MotionEvent.ACTION_MOVE: {
                Log.d(TAG, "MotionEvent.ACTION_MOVE");
                return true;
            }
            case MotionEvent.ACTION_UP: {
                Log.d(TAG, "MotionEvent.ACTION_UP");
                return true;
            }

        }
        return true;
    }

参考文章

View的绘制

看了大神们写的书,没有理解的很透,故记录在案:
View的工作原理/View的底层工作原理
自定义View/自定义控件


View类

ViewGroup

书中有句"ViewRoot对应于ViewRootImpl类"我当时理解为ViewRootImpl继承于ViewRoot,但是看过源码后,其实不是这样的,并不存在ViewRoot这个类。


ViewRootImpl

ViewRootImpl.java
    //执行遍历方法,贴出了主要代码片段,该方法中很长很长
    private void performTraversals() {
        //缓存mView,很多地方需要用到该对象
        final View host = mView;
        performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
        performLayout(lp, desiredWindowWidth, desiredWindowHeight);
        performDraw();

    }
    //查看performMeasure方法,这里贴出了全部的代码,该方法中调用了rmView的measure()方法。
    private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) {
        Trace.traceBegin(Trace.TRACE_TAG_VIEW, "measure");
        try {
            mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
        } finally {
            Trace.traceEnd(Trace.TRACE_TAG_VIEW);
        }
    }

测量流程--measure

测量完成后,计算出View的宽高。
测量的结果可通过getMeasureWidth(),getMeasureHeight()获得测量的宽高值。
MeasureSpec
DecorView的MeasureSpec产生过程
ViewRootImpl.java中的方法measureHierarchy()

 private boolean measureHierarchy(final View host, final WindowManager.LayoutParams lp,
            final Resources res, final int desiredWindowWidth, final int desiredWindowHeight) {
    childWidthMeasureSpec = getRootMeasureSpec(desiredWindowWidth, lp.width);
    childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height);
}

    private static int getRootMeasureSpec(int windowSize, int rootDimension) {
        int measureSpec;
        switch (rootDimension) {

        case ViewGroup.LayoutParams.MATCH_PARENT:
            // Window can't resize. Force root view to be windowSize.
            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY);
            break;
        case ViewGroup.LayoutParams.WRAP_CONTENT:
            // Window can resize. Set max size for root view.
            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST);
            break;
        default:
            // Window wants to be an exact size. Force root view to be that size.
            measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY);
            break;
        }
        return measureSpec;
    }

布局流程-layout

布局完成后会固定View的坐标及实际的宽高。
布局的结果可通过getLeft(),getTop(),getRight(), getBottom()获得布局的的左上右下的坐标。
getWidth(),getHeight()获取View的实际宽高值。

绘制流程-draw

Demo:滑块随着手指移动


滑块效果图

推荐阅读更多精彩内容