ViewGroup中事件传递机制

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

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

图片.png
A:dispatchTouchEvent 方法用于事件的分发,返回true,表示自己或者子view消费了该事件,不会向上冒泡,通过onInterceptTouchEvent进行判断是否拦截该事件。
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);
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容