优雅地实现RecycleView的点击、拖动、和侧滑删除

2024-08-22 15:08

本文主要是介绍优雅地实现RecycleView的点击、拖动、和侧滑删除,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

优雅地实现RecycleView的点击、拖动、和侧滑删除

自从发布了RecycleView之后,可以很方便的实现列表数据展示,同时只需要指定LayoutManager就可以实现列表、瀑布流、表格布局的无缝切换。RecycleView的所有东西都很美好,但是它没有提供像ListView那样的单击item的接口,都是通过对ViewHolder实现OnclickListener来实现单击事件。这里发现了一种更加优雅的方式,其本质是通过itemonTouchEvent事件来实现的。

点击事件

RecycleView 提供了一个接口叫做addOnItemTouchListener用来为每个item添加触摸事件,实现思路通过触摸手势监听,然后通过触摸坐标判断是哪个item

实现起来很简单:

1、定义一个OnItemClickListener继承自RecycleView.OnItemTouchListener,里面持有一个RecycleView和一个GestureDetectorCompat

public abstract class OnItemClickListener implements RecyclerView.OnItemTouchListener {private GestureDetectorCompat mGestureDetector;private RecyclerView recyclerView;public RecycleItemClickListener(RecyclerView view) {this.recyclerView = view;mGestureDetector = new GestureDetectorCompat(recyclerView.getContext(), ItemTouchHelperGestureListener());}

然后OnItemTouchListener有几个方法需要我们实现。onInterceptTouchEventonTouchEventonRequestDisallowInterceptTouchEvent

我们只需要在onInterceptTouchEventonTouchEvent中把事件交给mGestureDetector处理即可。

  @Overridepublic boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) {mGestureDetector.onTouchEvent(e);return false;}@Overridepublic void onTouchEvent(RecyclerView rv, MotionEvent e) {mGestureDetector.onTouchEvent(e);}

mGestureDetector捕捉到的事件都会交给ItemTouchHelperGestureListener来处理:

 class ItemTouchHelperGestureListener extends GestureDetector.SimpleOnGestureListener {@Overridepublic boolean onSingleTapUp(MotionEvent e) {View child = recyclerView.findChildViewUnder(e.getX(), e.getY());if (child != null) {int position = recyclerView.indexOfChild(child);onItemClick(recyclerView.getChildViewHolder(child), position);}return true;}@Overridepublic void onLongPress(MotionEvent e) {View child = recyclerView.findChildViewUnder(e.getX(), e.getY());if (child != null) {int position = recyclerView.indexOfChild(child);RecycleItemClickListener.this.onLongPress(recyclerView.getChildViewHolder(child), position);}}}

其中onItemClickonLongPressOnItemClickListener提供给外部调用的回调。其声明如下:

    public abstract void onItemClick(RecyclerView.ViewHolder holder, int position);public void onLongPress(RecyclerView.ViewHolder holder, int position) {}

onLongPress在需要的时候可以重写。

RecycleView使用时是这样的:

   recyclerView.addOnItemTouchListener(new RecycleItemClickListener(recyclerView) {@Overridepublic void onItemClick(RecyclerView.ViewHolder holder, int position) {Toast.makeText(SampleRecycleViewActivity.this, "点击了:" + position, Toast.LENGTH_SHORT).show();}@Overridepublic void onLongPress(RecyclerView.ViewHolder holder, int position) {}});

这样就实现了RecycleViewonItemClickListener了。

拖拽移动item

拖拽item的实现需要用到ItemTouchHelper这个接口,这个接口是SDK提供的用于处理拖拽、侧滑删除等功能的。

使用这个东西的步骤非常简单:

  1. 实例化一个ItemTouchHelper
  2. 关联RecycleView
    itemTouchHelper = new ItemTouchHelper(new ItemTouchHelper.Callback());itemTouchHelper.attachToRecyclerView(recyclerView);

就是这么简单。ItemTouchHelper的构造方法需要一个CallBack对象,这个CallBack对象,需要实现几个方法:getMovementFlagsonMoveonSwiped

其中getMovementFlags的实现如下:

            //设置移动方式@Overridepublic int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {final int dragFlags;final int swipeFlags;if (recyclerView.getLayoutManager() instanceof GridLayoutManager) {dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN | ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;swipeFlags = 0;} else {dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN;swipeFlags = ItemTouchHelper.START | ItemTouchHelper.END;}return makeMovementFlags(dragFlags, swipeFlags);}

上面代码中,dragFlags是用于拖拽的标志,swipeFlags是滑动标记。上面代码意思是如果是网格布局,拖动方向为上下左右,否则只有上下方向。swipeFlags是用于左右滑动的标志,这里暂且不说。

在设置了dragFlags的时候,就会在长按item的时候进入拖动模式,然后就会一直回调onMove函数。我们可以在onMove中进行数据集的更新:

            //移动过程中调用@Overridepublic boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {int fromPosition = viewHolder.getAdapterPosition();int toPosition = target.getAdapterPosition();data.add(toPosition, data.remove(fromPosition));adapter.notifyItemMoved(fromPosition, toPosition);return false;}

如果需要在拖拽过程中进行高亮背景切换,可以重写onSelectedChangedclearView方法。

            //当长按的时候调用@Overridepublic void onSelectedChanged(RecyclerView.ViewHolder viewHolder, int actionState) {if (actionState != ItemTouchHelper.ACTION_STATE_IDLE) {viewHolder.itemView.setBackgroundColor(Color.LTGRAY);}super.onSelectedChanged(viewHolder, actionState);}//当手指松开的时候调用@Overridepublic void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {super.clearView(recyclerView, viewHolder);viewHolder.itemView.setBackgroundResource(0);}

如果有需要对某些特定item不能拖动的话,可以重写isLongPressDragEnabled返回false,设置默认不可拖动。然后在前面实现的OnItemClickListeneronLongPress中对需要拖拽的item使用itemTouchHelper.startDrag(holder);就可以拖动了

滑动删除

首先设置标志位:

            //设置移动方式@Overridepublic int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {final int dragFlags;final int swipeFlags;if (recyclerView.getLayoutManager() instanceof GridLayoutManager) {dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN | ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;swipeFlags = 0;} else {dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN;swipeFlags = ItemTouchHelper.START | ItemTouchHelper.END;}return makeMovementFlags(dragFlags, swipeFlags);}

STARTEND的标志位是指可以左右滑动。

其次在onSwipe方法中进行数据集更新就可以了:

            //侧滑过程中调用@Overridepublic void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {int position = viewHolder.getAdapterPosition();adapter.notifyItemRemoved(position);data.remove(position);}

整体效果如下:

ezgif.com-video-to-gif.gif

这篇关于优雅地实现RecycleView的点击、拖动、和侧滑删除的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1096620

相关文章

MyBatis-Plus逻辑删除实现过程

《MyBatis-Plus逻辑删除实现过程》本文介绍了MyBatis-Plus如何实现逻辑删除功能,包括自动填充字段、配置与实现步骤、常见应用场景,并展示了如何使用remove方法进行逻辑删除,逻辑删... 目录1. 逻辑删除的必要性编程1.1 逻辑删除的定义1.2 逻辑删php除的优点1.3 适用场景2.

C#借助Spire.XLS for .NET实现在Excel中添加文档属性

《C#借助Spire.XLSfor.NET实现在Excel中添加文档属性》在日常的数据处理和项目管理中,Excel文档扮演着举足轻重的角色,本文将深入探讨如何在C#中借助强大的第三方库Spire.... 目录为什么需要程序化添加Excel文档属性使用Spire.XLS for .NET库实现文档属性管理Sp

Python+FFmpeg实现视频自动化处理的完整指南

《Python+FFmpeg实现视频自动化处理的完整指南》本文总结了一套在Python中使用subprocess.run调用FFmpeg进行视频自动化处理的解决方案,涵盖了跨平台硬件加速、中间素材处理... 目录一、 跨平台硬件加速:统一接口设计1. 核心映射逻辑2. python 实现代码二、 中间素材处

Java数组动态扩容的实现示例

《Java数组动态扩容的实现示例》本文主要介绍了Java数组动态扩容的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1 问题2 方法3 结语1 问题实现动态的给数组添加元素效果,实现对数组扩容,原始数组使用静态分配

Python实现快速扫描目标主机的开放端口和服务

《Python实现快速扫描目标主机的开放端口和服务》这篇文章主要为大家详细介绍了如何使用Python编写一个功能强大的端口扫描器脚本,实现快速扫描目标主机的开放端口和服务,感兴趣的小伙伴可以了解下... 目录功能介绍场景应用1. 网络安全审计2. 系统管理维护3. 网络故障排查4. 合规性检查报错处理1.

Python轻松实现Word到Markdown的转换

《Python轻松实现Word到Markdown的转换》在文档管理、内容发布等场景中,将Word转换为Markdown格式是常见需求,本文将介绍如何使用FreeSpire.DocforPython实现... 目录一、工具简介二、核心转换实现1. 基础单文件转换2. 批量转换Word文件三、工具特性分析优点局

Springboot3统一返回类设计全过程(从问题到实现)

《Springboot3统一返回类设计全过程(从问题到实现)》文章介绍了如何在SpringBoot3中设计一个统一返回类,以实现前后端接口返回格式的一致性,该类包含状态码、描述信息、业务数据和时间戳,... 目录Spring Boot 3 统一返回类设计:从问题到实现一、核心需求:统一返回类要解决什么问题?

Java使用Spire.Doc for Java实现Word自动化插入图片

《Java使用Spire.DocforJava实现Word自动化插入图片》在日常工作中,Word文档是不可或缺的工具,而图片作为信息传达的重要载体,其在文档中的插入与布局显得尤为关键,下面我们就来... 目录1. Spire.Doc for Java库介绍与安装2. 使用特定的环绕方式插入图片3. 在指定位

Java使用Spire.Barcode for Java实现条形码生成与识别

《Java使用Spire.BarcodeforJava实现条形码生成与识别》在现代商业和技术领域,条形码无处不在,本教程将引导您深入了解如何在您的Java项目中利用Spire.Barcodefor... 目录1. Spire.Barcode for Java 简介与环境配置2. 使用 Spire.Barco

Java利用Spire.Doc for Java实现在模板的基础上创建Word文档

《Java利用Spire.DocforJava实现在模板的基础上创建Word文档》在日常开发中,我们经常需要根据特定数据动态生成Word文档,本文将深入探讨如何利用强大的Java库Spire.Do... 目录1. Spire.Doc for Java 库介绍与安装特点与优势Maven 依赖配置2. 通过替换