完美解决Android RyclerView嵌套滑动事件冲突

御弟哥哥
• 阅读 1962

在Android项目开发中,为了实现需求和兼并用户体验,相信很多人都碰到滑动事件冲突的问题。在Android系统中事件分发机制是一个很重要的组成部分,由于这事件分发机制不是本文重点,故不在此多述,如果有想详细了解的可以自己搜下,网上有很多相关资料详细描述了Android事件分发机制。

一、问题场景

由于RecyclerView自身的优点,使得它已经基本取代了GridView、ListView,而且ViewPager2也是基于RecyclerView实现的,所以现在涉及到列表的基本都离不开RecyclerView。

本文就就基于项目中采用RecyclerView + ViewPager + Fragment + RecyclerView这种嵌套方式出现了滑动冲突。

完美解决Android RyclerView嵌套滑动事件冲突

QQ截图20200516115700.png

二、三种解决方式

首先讲下当下的几种处理方式:

  • 在父RecyclerView中的事件拦截事件中处理;
    自定义父recyclerView并重写onInterceptTouchEvent()方法,代码如下:

    public class ParentRecyclerView extends RecyclerView {
    
      public ParentRecyclerView(@NonNull Context context) {
          this(context,null);
      }
      public ParentRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs) {
          this(context, attrs,0);
      }
    
      public ParentRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyle) {
          super(context, attrs, defStyle);
      }
    
      //不拦截,继续分发下去
      @Override
      public boolean onInterceptTouchEvent(MotionEvent e) {
          //当然这里可能要根据实际场景去处理下,不仅仅是返回false就结束了。
         //todo : 实际场景处理代码
         //---------------------------------------------------------------------------------
          return false;
        }
    } 
  • 在子RecyclerView中的事件拦截事件中处理;
    通过requestDisallowInterceptTouchEvent方法干预事件分发过程,该方法就是通知父布局要不要拦截事件
    自定义子RecyclerView并重写dispatchTouchEvent,如下:

    public class ChildRecyclerView extends RecyclerView {

      public ChildRecyclerView (@NonNull Context context) {
          this(context,null);
      }

      public ChildRecyclerView (@NonNull Context context, @Nullable AttributeSet attrs) {
          this(context, attrs,0);
      }

      public ChildRecyclerView (@NonNull Context context, @Nullable AttributeSet attrs, int defStyle) {
          super(context, attrs, defStyle);
      }

      @Override
      public boolean dispatchTouchEvent(MotionEvent ev) {
          //父层ViewGroup不要拦截点击事件,true不要拦截,false拦截
          getParent().requestDisallowInterceptTouchEvent(true);
          return super.dispatchTouchEvent(ev);
      }
    } 
    ```

*   采用优先级最高的OnTouchListener;  
    从事件分发机制上看,OnTouchListener优先级很高,可以通过这个来告诉父布局,不要拦截我的事件
recyclerView.setOnTouchListener(new View.OnTouchListener() {
          @Override
          public boolean onTouch(View view, MotionEvent motionEvent) {
              switch (motionEvent.getAction()){
                  case MotionEvent.ACTION_DOWN:
                  case MotionEvent.ACTION_MOVE:
                      //这里有时要根据自己的场景去写自己的逻辑
                      view.getParent().requestDisallowInterceptTouchEvent(true);
                      break;
                  case MotionEvent.ACTION_UP:
                  case MotionEvent.ACTION_CANCEL:
                      view.getParent().requestDisallowInterceptTouchEvent(false);
                      break;
              }
              return true;
          }
      }); 
```

以上三种方式至于采用哪种要根据自己的实际场景。

三、针对一种方式进行详解

下面就针对第二种方式在自定义子RecyclerView的做事件拦截处理,因为这种方式正好适合项目解决冲突。

目标 :触摸子RecyclerView上下滑动时,子列表滑动,当列表滑动到顶部、底部或触摸点超出子RecyclerView上下边距时继续滑动,则父RecyclerView跟着滑动。
  • MotionEvent.ACTION_DOWN
    按下时记录按下的x,y值,并重置标记为;
    float x = ev.getX();
          float y = ev.getY();
          switch (ev.getAction()) {
              case MotionEvent.ACTION_DOWN:
                  mDownX = x;
                  mDownY = y;
                  lastY = y;
                  disallowInterceptState = 0;
                  getParent().requestDisallowInterceptTouchEvent(true);
                  break; 
  • MotionEvent.ACTION_MOVE
    手指滑动时,通过计算在x,y轴方向移动的距离,判断哪个方向先移动超过给的距离来判断移动的方向,若是x轴方向则不拦截(因为ViewPager横线滑动)次数将标记位设置为2(disallowInterceptState = 2),若y方向则告诉父View不要拦截并将标记位设置为1(disallowInterceptState = 1);
    继续move时,不断检查是否到View的上下边缘和列表是否滑动到顶部或底部,当满足条件时将标记位设置为2,(disallowInterceptState = 2)告诉父View可以拦截事件了。
    if (disallowInterceptState == 0) {
         float absX = Math.abs(x - mDownX);
         float absY = Math.abs(y - mDownY);
         if ((absX > 5f || absY > 5f)) {
           if (absX < absY) {
              disallowInterceptState = 1;
           } else {
              disallowInterceptState = 2;
            }
        }
     }
    if (getParent() != null && disallowInterceptState != 0) {
              //y坐标边界检测
              boolean bl = y < 0 || y > getMeasuredHeight();
              disallowInterceptState = bl ? 2 : disallowInterceptState;
              //若滑动到顶部 && 继续下滑动,则释放拦截事件
              if((isScrollTop() && lastY < y) || (isScrollBottom() && lastY > y)){
                 disallowInterceptState = 2;
              }
           //检查滑动到底部或顶部
           getParent().requestDisallowInterceptTouchEvent(disallowInterceptState == 1);
       }
    lastY = y; 
  • MotionEvent.ACTION_UP和MotionEvent.ACTION_CANCEL
    这两个事件不需要做其他处理,恢复父view可以拦截事件
    //父层ViewGroup不要拦截点击事件
    getParent().requestDisallowInterceptTouchEvent(false); 

完整代码ChildRecyclerView.java

public class ChildRecyclerView extends RecyclerView {

    public ChildRecyclerView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    private float mDownX, mDownY,lastY;
    private int disallowInterceptState = 0;

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        float x = ev.getX();
        float y = ev.getY();
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mDownX = x;
                mDownY = y;
                lastY = y;
                disallowInterceptState = 0;
                getParent().requestDisallowInterceptTouchEvent(true);
                break;
            case MotionEvent.ACTION_MOVE:
                if (disallowInterceptState == 0) {
                    float absX = Math.abs(x - mDownX);
                    float absY = Math.abs(y - mDownY);
                    if ((absX > 5f || absY > 5f)) {
                        if (absX < absY) {
                            disallowInterceptState = 1;
                        } else {
                            disallowInterceptState = 2;
                        }
                    }
                }
                if (getParent() != null && disallowInterceptState != 0) {
                    //y坐标边界检测
                    boolean bl = y < 0 || y > getMeasuredHeight();
                    disallowInterceptState = bl ? 2 : disallowInterceptState;
                    //若滑动到顶部 && 继续下滑动,则释放拦截事件
                    if((isScrollTop() && lastY < y) || (isScrollBottom() && lastY > y)){
                        disallowInterceptState = 2;
                    }
                    //检查滑动到底部或顶部
                    getParent().requestDisallowInterceptTouchEvent(disallowInterceptState == 1);
                }
                lastY = y;
                break;
            case MotionEvent.ACTION_UP:
            case MotionEvent.ACTION_CANCEL:
                //父层ViewGroup不要拦截点击事件
                getParent().requestDisallowInterceptTouchEvent(false);
                break;
        }

        return super.dispatchTouchEvent(ev);
    }


    /**
     * 滑动到底部检查
     * @return true滑动到底部,false没有到底
     */
    private boolean isScrollBottom(){
        return !canScrollVertically(1);
    }

    /**
     * 滑动到顶部检查
     * @return true滑动到顶部,false没有到顶
     */
    private boolean isScrollTop(){
        return !canScrollVertically(-1);
    }
} 

最后附上效果图

完美解决Android RyclerView嵌套滑动事件冲突

效果图.gif

希望能帮助到大家。

每日一句:要想练就绝世武功 就要忍受常人难忍受的痛。

本文转自 https://www.jianshu.com/p/2cadf44a1448,如有侵权,请联系删除。

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
九路 九路
2年前
一文读懂Android View事件分发机制
AndroidView虽然不是四大组件,但其并不比四大组件的地位低。而View的核心知识点事件分发机制则是不少刚入门同学的拦路虎。ScrollView嵌套RecyclerView(或者ListView)的滑动冲突这种老大难的问题的理论基础就是事件分发机制。事件分发机制面试也会经常被提及,如果你能get到要领,并跟面试官深入的灵魂交流一下,那么一定会让
刘望舒 刘望舒
2年前
Android输入系统(一)输入事件传递流程和InputManagerService的诞生
Android框架层Android输入系统Android框架层本文首发于微信公众号「刘望舒」前言很多同学可能会认为输入系统是不是和View的事件分发有些关联,确实是有些关联,只不过View事件分发只能算是输入系统事件传递的一部分。这个系列讲的输入系统主要是我们不常接触的,但还是需要去了解的那部分。1.输入事件传递流程的组成部分输入系统是外界与And
刘望舒 刘望舒
2年前
Android输入系统(四)输入事件是如何分发到目标窗口的?
Android框架层Android输入系统Android框架层本文首发于微信公众号「刘望舒」基于Android8.1前言在这篇文章中,由于文章篇幅的原因,InputDispatcher的分发过程还有一部分没有讲解,这一部分就是事件分发到目标窗口的过程。1.为事件寻找合适的分发目标我们先来回顾上一篇文章讲解的InputDispatcher的disp
Android事件分发-基础原理和场景分析
和其他平台类似,Android中View的布局是一个树形结构,各个ViewGroup和View是按树形结构嵌套布局的,从而会出现用户触摸的位置坐标可能会落在多个View的范围内,这样就不知道哪个View来响应这个事件,为了解决这一问题,就出现了事件分发机制。
Stella981 Stella981
2年前
Android Hook技术
1\.什么是HookHook英文翻译过来就是「钩子」的意思,那我们在什么时候使用这个「钩子」呢?在Android操作系统中系统维护着自己的一套事件分发机制。应用程序,包括应用触发事件和后台逻辑处理,也是根据事件流程一步步地向下执行。而「钩子」的意思,就是在事件传送到终点前截获并监控事件的传输,像个钩子钩上事件一样,并且能够在钩上事件时
Easter79 Easter79
2年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Stella981 Stella981
2年前
Android蓝牙连接汽车OBD设备
//设备连接public class BluetoothConnect implements Runnable {    private static final UUID CONNECT_UUID  UUID.fromString("0000110100001000800000805F9B34FB");
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这