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

御弟哥哥 等级 886 0 0

在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,如有侵权,请联系删除。

收藏
评论区

相关推荐

RecyclerView之ItemDecoration使用教程
译文的GitHub地址:RecyclerView之ItemDecoration由浅入深(https://link.jianshu.com?thttps://github.com/thinkSky1206/androidblog/blob/master/RecyclerView%E4%B9%8BItemDecoration%E7%94%B1%E6%B5
Android RecyclerView如何获取滑动距离
获取RecyclerView滑动的距离。 本文演示如何获取RecyclerView的滑动距离。 要实现这个功能,需要给RecyclerView添加滑动时监听RecyclerView.OnScrollListener。 recyclerView.addOnScrollListener(new RecyclerView.OnScrollListene
完美解决Android RyclerView嵌套滑动事件冲突
在Android项目开发中,为了实现需求和兼并用户体验,相信很多人都碰到滑动事件冲突的问题。在Android系统中事件分发机制是一个很重要的组成部分,由于这事件分发机制不是本文重点,故不在此多述,如果有想详细了解的可以自己搜下,网上有很多相关资料详细描述了Android事件分发机制。 一、问题场景 由于RecyclerView自身的优点,使得它已经基本
干货|详解位图算法在Android RecyclerView中的应用
1. 前言 1.1 关于算法金庸武侠小说中的主人公在成为绝世高手之前,都会学习一门玄门内功。郭靖有了全真派的内功才能修炼九阴真经、虚竹得到了无崖子的毕生功力后,武学造诣日渐精进、张无忌苦练五年九阳神功,日后才能融合乾坤大挪移。对于程序员,算法就是小说中的内功,编程语言就是不同门派的武功。张无忌因为有九阳神功加持仅用一天就学会了阳顶天几十年都学不成的乾
RecyclerView基础用法
是一款非常强大的 widget,它可以帮助您灵活地显示列表数据。当我开始学习 RecyclerView 的时候,我发现对于复杂的列表界面有很多资源可以参考,但是对于简单的列表展现就鲜有可参考的资源了。虽然 RecyclerView 的组成结构乍一看有些复杂,但是深入理解以后您会发现它其实非常简单明了。本文会通过创建一个简单的 RecyclerView 实现一
RecyclerView更全解析之 - 为它优雅的添加头部和底部
1.概述 上一期的,解决了几个坑。那么这一期我们来动态为RecyclerView去加载头部和底部,为上一期的RecyclerView列表数据添加广告轮播图,至于广告轮播大家可以看一下这一期 ,这里我就不多讲了,直接拿过来用。      视频讲解:相关文章:                                    
Java后端部署以及与Android通信注意事项
1 概述 ==== 本文列举了一些`Android`+后端`Java`通信/部署时的问题以及注意事项,覆盖的问题包括但不限于安全组、数据库、路径等,如果各位读者的`Android`端不能正常访问`Java`后端,希望这里的解决方案能帮助到您。 2 分类 ==== 这里将问题分为三类: * `Java`端问题 * `Android`端问题 *
java版本springcloud+springboot+mybatis 分布式 微服务 多租户 电子商务 直播带货 短视频带货 社交电商平台
涉及平台:平台管理(包含自营店面)、商家端(PC端、手机端)、买家平台(PC端、H5/公众号、小程序、APP端(IOS/Android)、微服务平台(业务服务) 核心架构:Spring Cloud、Spring Boot、Mybatis、Redis、SFTP 前端框架:VUE、Uniapp、Bootstrap/H5/CSS3、IOS、Android、小程
java编程中使用二进制进行权限或状态控制
直接看代码以及注释吧。 @Test public void main() { // PC WEB端 int pc = 1 << 0;// ...0001=1 // Android端 int android = 1 <<
Android RecyclerView使用GridLayoutManager间距设置
使用RecyclerView设置间距,需要重写RecyclerView.ItemDecoration这个类。有如下的效果图需要实现,间距只有中间的格子和底部的格式之间有。 Paste\_Image.png 实现方法很简单,因为这个效果是每一行有3个格子,只要每行的第一个格式左边间距为0即可以。其他都设置左边距和底部距离。 代码如下: publ
Android 解决NestedScrollView 嵌套 RecyclerView出现的卡顿,上拉刷新无效
解决卡顿的方法最简单的就是设置RecyclerView的android:nestedScrollingEnabled="false",放弃自己的滑动,交给外部的NestedScrollView处理,就没有出现卡顿的现象了 至于RecyclerView的上拉刷新,可以监听NestedScrollView的滑动监听,具体代码如下: mNestedSc
Android+Spring Boot 选择+上传+下载文件
2021.02.03更新 ============ 1 概述 ==== 前端`Android`,上传与下载文件,使用`OkHttp`处理请求,后端使用`Spring Boot`,处理`Android`发送来的上传与下载请求。这个其实不难,就是特别多奇奇怪怪的坑,因此,就一句话, 希望各位读者能少走弯路。 2 环境 ==== * `Win10` *
Android学习系列笔记(五)
##Android基础网络第二天 1 post方式提交数据乱码的解决 ================= 一般在公司开发客户端和服务端的编码要保持一致。 android端的默认编码是utf-8; 做url请求时需要对参数进行URLEncode编码. URL url = new URL("http://1
Android端Charles抓包
#### 目录介绍 * 01.下载安装 * 02.抓包代理设置 * 03.抓包Https操作 * 04.抓包原理介绍 * 05.抓包数据介绍 * 06.常见问题总结 * 07.Android拦截抓包 ### 01.下载安装 * 下载地址(下载对应的平台软件即可) * [https://www.charlesp
RecyclerView实现倒序列表
RecyclerView实现倒序列表 ================== 标签(空格分隔): android RecyclerView 倒序 * * * ##1、写在前面 实现一个聊天界面,就是类似QQ那种,这里是讲一下倒序排列,不实现QQ的各种高级功能 ##2、ListView 反转数据 只要把数据倒序加入到adapter的数据集中,就可以实现倒