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

御弟哥哥 等级 420 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
js动态生成二维码
需求:项目需要根据链接实时生成二维码,当检测终端是PC时,将当前项目链接生成二维码供用户手机端使用 判断终端是否为mobile function isMobile () { let flag navigator.userAgent.match(/(phone|pad|pod|iPhone|iPod|ios|iPad|Android|Mobile
Android RecyclerView如何获取滑动距离
获取RecyclerView滑动的距离。 本文演示如何获取RecyclerView的滑动距离。 要实现这个功能,需要给RecyclerView添加滑动时监听RecyclerView.OnScrollListener。 recyclerView.addOnScrollListener(new RecyclerView.OnScrollListene
Android深入浅出之Binder机制
Android深入浅出之Binder机制 一 说明 Android系统最常见也是初学者最难搞明白的就是Binder了,很多很多的Service就是通过Binder机制来和客户端通讯交互的。所以搞明白Binder的话,在很大程度上就能理解程序运行的流程。 我们这里将以MediaService的例子来分析Binder的使用: ServiceMan
完美解决Android RyclerView嵌套滑动事件冲突
在Android项目开发中,为了实现需求和兼并用户体验,相信很多人都碰到滑动事件冲突的问题。在Android系统中事件分发机制是一个很重要的组成部分,由于这事件分发机制不是本文重点,故不在此多述,如果有想详细了解的可以自己搜下,网上有很多相关资料详细描述了Android事件分发机制。 一、问题场景 由于RecyclerView自身的优点,使得它已经基本
移动端H5开发常用技巧总结
html 篇 常用的meta属性设置 meta对于移动端的一些特殊属性,可根据需要自行设置 <meta name"screenorientation" content"portrait" //Android 禁止屏幕旋转 <meta name"fullscreen" content"yes"             //全屏显示
Android开发 常见异常和解决办法(一)
Android Studio是Android开发的理想工具,但是由于版本的更新和配置的差异,会出现很多问题,下面是以《第一行代码 第二版》为基础进行开发学习可能遇见的一些问题及其解决办法。 1.Android Studio 3.0及以上版本找不到Android Device Monitor: 解决办法: (1)在Android Studio中打开终端,如下
干货|详解位图算法在Android RecyclerView中的应用
1. 前言 1.1 关于算法金庸武侠小说中的主人公在成为绝世高手之前,都会学习一门玄门内功。郭靖有了全真派的内功才能修炼九阴真经、虚竹得到了无崖子的毕生功力后,武学造诣日渐精进、张无忌苦练五年九阳神功,日后才能融合乾坤大挪移。对于程序员,算法就是小说中的内功,编程语言就是不同门派的武功。张无忌因为有九阳神功加持仅用一天就学会了阳顶天几十年都学不成的乾
Django+Vue开发生鲜电商平台之4.Restful API和Vue介绍
也许今天你是最好的,但未必明天还最好;今天也许你是最差的,但社会给了你很多的机会,只要你把握,只要努力,总会有机会。 ——马云Github和Gitee代码同步更新:;。后端架构搭建好之后,需要搭建前端架构。 一、Restful API介绍 1.前后端分离优缺点近年来,随着多种平台类型(PC端、Android端、Mac端、iPhone端、P
Android 世界中,谁喊醒了 Zygote ?
Zygote 作为 Android 世界的受精卵,在成功繁殖出 system_server 进程之后并没有完全功成身退,仍然承担着受精卵的责任。Zygote 通过调用其持有的 ZygoteServer 对象的 runSelectLoop() 方法开始等待客户端的呼唤,有求必应。客户端的请求无非是创建应用进程,以 startActivit
RecyclerView基础用法
是一款非常强大的 widget,它可以帮助您灵活地显示列表数据。当我开始学习 RecyclerView 的时候,我发现对于复杂的列表界面有很多资源可以参考,但是对于简单的列表展现就鲜有可参考的资源了。虽然 RecyclerView 的组成结构乍一看有些复杂,但是深入理解以后您会发现它其实非常简单明了。本文会通过创建一个简单的 RecyclerView 实现一
Flutter 跨平台演进及架构开篇
版权声明: 本站所有博文内容均为原创,转载请务必注明作者与原文链接,且不得篡改原文内容。一、移动跨平台技术演进 1\. 引言移动互联网发展十余年,伴随着 Android、iOS 等智能手机的不断普及,移动端已逐步取代 PC 端,成为兵家必争之地。正所谓“得移动端者得天下”,移动端已成为互联网领域最大的流量分发入口,一
Android AOSP基础(一)VirtualBox 安装 Ubuntu
AOSP基础 Android框架层本文首发于微信公众号「刘望舒」 前言在Android进阶三部曲第二部《Android进阶解密》的第一章,我介绍了两种阅读源码的方式,其中一种是从百度网盘:https://pan.baidu.com/s/1ngsZs 将源码下载下来,然后用SouceInsight来查看,这种方式很便捷,适合去阅读源码,但是有两个弊端,一个是无
RecyclerView更全解析之 - 仿支付宝侧滑删除和拖动排序
1.概述 这是春节前的最后一篇分享技术的博客了,接下来的时间需要去完善视频讲解,至于今年都干了哪些事有什么成就吹牛的这里就不多说了,声明一下图片资源我是盗用的别人的。这是最后一期分享RecyclerView了,我们直接看这一期需要分享的效果:      这里写图片描述         视频讲解:相关文章:               
RecyclerView更全解析之 - 为它优雅的添加头部和底部
1.概述 上一期的,解决了几个坑。那么这一期我们来动态为RecyclerView去加载头部和底部,为上一期的RecyclerView列表数据添加广告轮播图,至于广告轮播大家可以看一下这一期 ,这里我就不多讲了,直接拿过来用。      视频讲解:相关文章: