ViewGroup中事件传递机制

10 阅读5分钟

【Android】View的事件分发机制_view的分发事件-CSDN博客

1.ViewGroup事件的消耗和传递主要是通过以下3个方法实现

图片.png

dispatchTouchEvent 的返回值决定了事件的后续流向。其含义必须根据调用者是谁来精确理解

返回值当调用者是 Activity 时当调用者是 ViewGroup 或 View 时
return true事件被消费,终止传递。 Activity 自身消费了事件,不会调用 Window 的 super.dispatchTouchEvent后续所有 View 都收不到此事件事件在本层被消费,终止向下分发和向上冒泡。 • ViewGroup: 不会调用 onInterceptTouchEvent 和子View的 dispatchTouchEvent。 • View: 不会调用自身的 onTouchEvent
return false事件未被消费,继续传递。 Activity 会调用 Window 的 super.dispatchTouchEvent,将事件传递给底层 View 树。事件未被本层消费,向上级报告。 • ViewGroup情况复杂,需分是否拦截讨论。 • View: 表示自身的 onTouchEvent 返回了 false,即未消费事件,事件将回传给父容器的 onTouchEvent
return super.dispatchTouchEvent(event)标准传递。  将事件交给系统默认机制处理,即向下传递。执行默认逻辑。 • ViewGroup: 会触发 onInterceptTouchEvent,根据其返回值决定是否分发给子View。 • View: 会调用自身的 onTouchEvent,并以其返回值作为本方法返回值
A:dispatchTouchEvent 方法用于事件的分发,返回true,表示自己或者子view消费了该事件,不会向上冒泡,返回false,不拦截,事件可以向下传递.
B:onInterceptTouchEvent是 ViewGroup中特有的方法,View中没有,它的作用负责拦截事件,返回true的时候,表示拦截当前事件,不继续往下分发,交给自身onTouchEvent进行处理(如果是false,自身的onTouchEvent不会触发,其他的子控件的onTouchEvent会消费该事件)。返回false则不拦截,继续往下传递,这是ViewGroup特有的方法,因为ViewGroup中可能还有子View,而在Android中View中是不能再包含子View的。
C:onTouchEvent方法用于事件的处理,返回true表示消费当前事件,返回false则不处理,交给子控件继续分发。
    1.ViewGroup包含这三个方法,而View则只包含dispatchTouchEvent()、onTouchEvent()两个方法,不 
    包含onInterceptTouchEvent()。

    2.触摸事件由Action_Down、Action_Move、Action_Up组成,一次完整的触摸事件,包含一个Down和 
       Up,以及若干个Move(可以为0);

    3.在Action_Down的情况下,事件会先传递到最顶层的ViewGroup,调用ViewGroup的
     dispatchTouchEvent():a)如果ViewGroup的onInterceptTouchEvent()返回false不拦截该事件,
    则会分发给子View,调用子View的dispatchTouchEvent(),如果子View的dispatchTouchEvent()返回 
    true,则调用View的onTouchEvent()消费事件;b)如果ViewGroup的onInterceptTouchEvent()
    返回true拦截该事件,则调用ViewGroup的onTouchEvent()消费事件,接下来的Move和Up事件将由该 
    ViewGroup直接进行处理。

    4.当某个子View的dispatchTouchEvent()返回true时,会中止Down事件的分发,
       同时在ViewGroup中记录该子View。接下来的Move和Up事件将由该子View直接进行处理。

    5.当ViewGroup中所有子View都不捕获Down事件时,将触发ViewGroup自身的onTouch();
     触发的方式是调用super.dispatchTouchEvent函数,即父类View的dispatchTouchEvent方法。
     在所有子View都不处理的情况下,触发Acitivity的onTouchEvent方法。

    6.由于子View是保存在ViewGroup中的,多层ViewGroup的节点结构时,
       上层ViewGroup保存的会是真实处理事件的View所在的ViewGroup对象。
       如ViewGroup0——ViewGroup1——TextView的结构中,TextView返回了true,它将被保存在 
      ViewGroup1中,而ViewGroup1也会返回true,将被保存在ViewGroup0中;当Move和Up事件来时, 
      会先从ViewGroup0传递到ViewGroup1,再由ViewGroup1传递到TextView,最后事件由TextView消 费掉。

    7.子View可以调getParent().requestDisallowInterceptTouchEvent(),请求父ViewGroup不拦截事件。

图片.png

2.ViewPager里面嵌套了二个Fragment,FragmentA布局如下:

上面3个Progerssbar,下面一个左右的滑动的HorizontalScrollView,活动到右边的时候,导致无法切换tab页面,现在提供 二种思路解决: #####1.通过Viewpager的onInterceptTouchEvent()方法,在HorizontalScrollView滑动 到最右边的时候,返回true,拦截 该事件,自己消费,这样就可以左右滑动,切换tab了

package com.example.administrator.myapplication;

import android.annotation.TargetApi;
import android.content.Context;
import android.os.Build;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.widget.HorizontalScrollView;

/**
 * Created by Administrator on 2018/2/14.
 */

public class MyViewPager extends ViewPager {
    private HorizontalScrollView scrollview=null;
    private float downX;
    private float downY;
    private float endY;
    private float endX;
    boolean result=false;
    public MyViewPager(Context context) {
        super(context);
    }

    public MyViewPager(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    /***
     * 如果当前事件返回true,拦截事件,将会触发当前控件的onTouchEvent方法
       如果当前方法,返回false ,不拦截事件,事件继续传递给孩子(子视图)
     * @param ev
     * @return
     */

    @TargetApi(Build.VERSION_CODES.M)
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        switch (ev.getAction()){
            case MotionEvent.ACTION_DOWN:
               downX=ev.getX();
                downY=ev.getY();
                break;
            case MotionEvent.ACTION_MOVE:
                endX= (int) ev.getX();
                endY=ev.getY();
                float distanceX=Math.abs(endX-downX);
                float distanceY=Math.abs(endY-downY);
                if(distanceX>distanceY&&distanceX>10){
                    result=true;
                }
                break;
            case MotionEvent.ACTION_UP:
                break;
        }
        Log.i("TAG","onInterceptTouchEvent:"+result);
        return result;
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        switch (ev.getAction()){
            case MotionEvent.ACTION_DOWN:
                Log.i("TAG","ACTION_DOWN");
                break;
            case MotionEvent.ACTION_MOVE:
                Log.i("TAG","ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                Log.i("TAG","ACTION_UP");
                break;
        }

        return super.onTouchEvent(ev);
    }
}

图片.png

2.通过重写HorizontalScrollView的onTouchEvent()方法,在HorizontalScrollView滑动 到最右边的时候,请求父View(ViewPager)拦截onTouchEvent事件,不继续向下传递
package com.example.administrator.myapplication;

import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.widget.HorizontalScrollView;

import static android.content.ContentValues.TAG;


/**
 * Created by Administrator on 2018/2/14.
 */

public class HorizontalView extends HorizontalScrollView {

    public HorizontalView(Context context) {
        super(context);
    }

    public HorizontalView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public HorizontalView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }



    /**
     * 第一个参数为变化后的X轴位置

     第二个参数为变化后的Y轴的位置

     第三个参数为原先的X轴的位置

     第四个参数为原先的Y轴的位置
     * @param l
     * @param t
     * @param oldl
     * @param oldt
     */
    @Override
    protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        super.onScrollChanged(l, t, oldl, oldt);

    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        switch (ev.getAction()){
            case MotionEvent.ACTION_DOWN:
                break;
            case MotionEvent.ACTION_MOVE:
                //请求Viewpager拦截Touch事件
                getParent().getParent().requestDisallowInterceptTouchEvent(false);
                Log.i(TAG,"onTouchEvent:ACTION_MOVE");
                break;
            case MotionEvent.ACTION_UP:
                break;
        }
        return super.onTouchEvent(ev);
    }
}