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

御弟哥哥
• 阅读 1629

在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
1年前
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
Stella981 Stella981
1年前
Android Hook技术
1\.什么是HookHook英文翻译过来就是「钩子」的意思,那我们在什么时候使用这个「钩子」呢?在Android操作系统中系统维护着自己的一套事件分发机制。应用程序,包括应用触发事件和后台逻辑处理,也是根据事件流程一步步地向下执行。而「钩子」的意思,就是在事件传送到终点前截获并监控事件的传输,像个钩子钩上事件一样,并且能够在钩上事件时
Easter79 Easter79
1年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Stella981 Stella981
1年前
Android蓝牙连接汽车OBD设备
//设备连接public class BluetoothConnect implements Runnable {    private static final UUID CONNECT_UUID  UUID.fromString("0000110100001000800000805F9B34FB");
3A网络 3A网络
4个月前
开发一个不需要重写成 Hive QL 的大数据 SQL 引擎
开发一个不需要重写成HiveQL的大数据SQL引擎学习大数据技术的核心原理,掌握一些高效的思考和思维方式,构建自己的技术知识体系。明白了原理,有时甚至不需要学习,顺着原理就可以推导出各种实现细节。各种知识表象看杂乱无章,若只是学习
SPDK对接Ceph性能优化
关键词:SPDK、NVMeOF、Ceph、CPU负载均衡SPDK是intel公司主导开发的一套存储高性能开发套件,提供了一组工具和库,用于编写高性能、可扩展和用户态存储应用。它通过使用一些关键技术实现了高性能:1.将所有必需的驱动程序移到用户空间,以避免系统调用并且支持零拷贝访问2.IO的完成通过轮询硬件而不是依赖中断,以降低时延3.使用消息传递,以避免IO
3A网络 3A网络
4个月前
理解 virt、res、shr 之间的关系(linux 系统篇)
理解virt、res、shr之间的关系(linux系统篇)前言想必在linux上写过程序的同学都有分析进程占用多少内存的经历,或者被问到这样的问题——你的程序在运行时占用了多少内存(物理内存)?通常我们可以通过t
SPDK QOS机制解析
本文关键词:intelspdkbdevqos序:intelspdk软件在存储领域应用广泛。因其可以高效管理linux系统的nvmessd盘,又支持vhostuser协议可以对接qemu虚拟机,在云计算领域通常被用来做本地盘云主机的存储管理软件。如此优秀的一款软件,有必要仔细分析其内部的实现机制,本篇文章主要介绍spdkqos机制。spdk
一个关于SDWAN单臂部署方案验证的实验
假设有这样一张网络,其中RTA和PCA表示某公司的A分支,通过中国电信CT路由器接入互联网ISP;RTB和PCB表示某公司的B分支,通过中国联通CU路由器接入互联网ISP。DNS(8.8.8.8)表示某互联网应用。为实现A分支私网192.168.2.0/24和B分支私网192.168.3.0/24的互通,现计划使用某厂商的SDWAN方案进打通两个内网,像下图
高性能API网关Kong介绍
本文关键词:高性能、API网关、Kong、微服务1.Introduction是随着微服务(Microservice)概念兴起的一种架构模式。原本一个庞大的单体应用(Allinone)业务系统被拆分成许多微服务(Microservice)系统进行独立的维护和部署,服务拆分带来的变化是API的规模成倍增长,API的管理难度也在日益增加,使用API网关发布和管
天翼云高可用虚拟IP(HAVIP)实践
(一)产品概述天翼云高可用虚拟IP(HighAvailabilityVirtualIPAddress,简称HAVIP)是一种可用独立创建和删除的私有网络IP地址资源。通过在VIPCIDR中申请一个私有网络IP地址,然后与高可用软件(如高可用软件Keepalived)配合使用,可用在VPC中搭建高可用的主备集群服务,提高VPC中服务的可用性。限制和说明