1,在实际项目中我们常常有对一个列表进行滑删除操作,使用我们昨天的ItemTouchHelper其实也可以实现简单的实现这个功能,先来看一下使用ItemTouchHelper来实现的效果:

2,从上面的效果图我们可以看到,大致的实现了我们的需求,具体操作如下

  第一步 :添加表示为START和END标识位

  1. @Override
  2. public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {
  3. if (recyclerView.getLayoutManager() instanceof GridLayoutManager) {
  4. final int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN |
  5. ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;
  6. final int swipeFlags = 0;
  7. return makeMovementFlags(dragFlags, swipeFlags);
  8. } else {
  9. final int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN;
  10.  
  11. final int swipeFlags = ItemTouchHelper.START | ItemTouchHelper.END;
  12. return makeMovementFlags(dragFlags, swipeFlags);
  13. }
  14. }

  第二步 : 在侧滑过程中删除数据

  1. @Override
  2. public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {
  3. int position = viewHolder.getAdapterPosition();
  4. myAdapter.notifyItemRemoved(position);
  5. datas.remove(position);
  6. }

  这样就很简单的实现了我们的操作

3,但是上面实现方法有一个短板,就是用户有时候是不小心向左滑动,这样我们最好在滑动后添加一个确定操作的按钮,以确保用户不是手误滑动,这是我们要自定义RecyclerView重写OnTouchEvent方法来实现,先看一下实现的效果:

  来说一下整体的思路

  第一步:RecyclerView的item布局是由两部分组成,一是我们正常的item,还有一部分是我们还有删除按钮的布局文件,布局文件展示效果如下:

代码如下:

item_linear.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="wrap_content"
  5. android:orientation="horizontal"
  6. android:id="@+id/ll_item"
  7. >
  8.  
  9. <LinearLayout
  10. android:layout_width="match_parent"
  11. android:layout_height="match_parent">
  12.  
  13. <TextView
  14. android:id="@+id/tv_title"
  15. android:layout_width="wrap_content"
  16. android:layout_height="wrap_content"
  17. android:layout_gravity="center_vertical"
  18. android:layout_marginLeft="5dp"
  19. android:layout_weight="1"
  20. android:text="我是标题"
  21. android:textSize="16dp"/>
  22.  
  23. <ImageView
  24. android:id="@+id/img"
  25. android:layout_width="wrap_content"
  26. android:layout_height="wrap_content"
  27. android:layout_marginBottom="5dp"
  28. android:layout_marginTop="5dp"
  29. android:src="@mipmap/ic_category_0"
  30. />
  31. </LinearLayout>
  32.  
  33. <!-- 屏幕右侧外边部分,正常时在屏幕中处于不可见 -->
  34. <LinearLayout
  35. android:id="@+id/ll_hidden"
  36. android:layout_width="50dp"
  37. android:layout_height="match_parent"
  38. android:background="#ff0000"
  39. android:gravity="center"
  40. >
  41.  
  42. <TextView
  43. android:id="@+id/tv_item_delete"
  44. android:layout_width="wrap_content"
  45. android:layout_height="wrap_content"
  46. android:text="删除"
  47. android:textColor="#ffffff"
  48. android:textSize="16sp"
  49. />
  50. </LinearLayout>
  51. </LinearLayout>

  第二步:创建自定义的RecyclerView,重写onTouchEvent()方法,而重写OnTouchEvent()的大致思路是当用户手指按下时,计算出当前选中的是哪个Item,并获取到该Item对象;然后判断手指移动方向,若左移,则滑动(在滑动之前,先恢复上次的状态);若右移,则恢复;当左移完成之后,“删除”按钮自然就“暴露”在屏幕上可点击的范围了;然后就可以对Item进行删除操作了。 由于代码中的注释很详细了,就不再讲解了

  SwipRecyclerView.java

  1. package com.qianmo.dragrecyclerview;
  2.  
  3. import android.content.Context;
  4. import android.graphics.Rect;
  5. import android.support.v7.widget.LinearLayoutManager;
  6. import android.support.v7.widget.RecyclerView;
  7. import android.util.AttributeSet;
  8. import android.util.Log;
  9. import android.view.MotionEvent;
  10. import android.view.View;
  11. import android.view.ViewConfiguration;
  12. import android.view.animation.LinearInterpolator;
  13. import android.widget.LinearLayout;
  14. import android.widget.Scroller;
  15. import android.widget.TextView;
  16.  
  17. /**
  18. * Created by Administrator on 2017/3/14 0014.
  19. * E-Mail:543441727@qq.com
  20. */
  21.  
  22. public class SwipeRecyclerView extends RecyclerView{
  23.  
  24. private static final String TAG = "RecycleView";
  25. private int maxLength, mTouchSlop;
  26. private int xDown, yDown, xMove, yMove;
  27. /**
  28. * 当前选中的item索引(这个很重要)
  29. */
  30. private int curSelectPosition;
  31. private Scroller mScroller;
  32.  
  33. private LinearLayout mCurItemLayout, mLastItemLayout;
  34. private LinearLayout mLlHidden;//隐藏部分
  35. private TextView mItemContent;
  36. private LinearLayout mItemDelete;
  37.  
  38. /**
  39. * 隐藏部分长度
  40. */
  41. private int mHiddenWidth;
  42. /**
  43. * 记录连续移动的长度
  44. */
  45. private int mMoveWidth = 0;
  46. /**
  47. * 是否是第一次touch
  48. */
  49. private boolean isFirst = true;
  50. private Context mContext;
  51.  
  52. /**
  53. * 删除的监听事件
  54. */
  55. private OnRightClickListener mRightListener;
  56.  
  57. public void setRightClickListener(OnRightClickListener listener){
  58. this.mRightListener = listener;
  59. }
  60.  
  61. public SwipeRecyclerView(Context context) {
  62. this(context, null);
  63. }
  64.  
  65. public SwipeRecyclerView(Context context, AttributeSet attrs) {
  66. this(context, attrs, 0);
  67. }
  68.  
  69. public SwipeRecyclerView(Context context, AttributeSet attrs, int defStyle) {
  70. super(context, attrs, defStyle);
  71. mContext = context;
  72. //滑动到最小距离
  73. mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
  74. //滑动的最大距离
  75. maxLength = ((int) (180 * context.getResources().getDisplayMetrics().density + 0.5f));
  76. //初始化Scroller
  77. mScroller = new Scroller(context, new LinearInterpolator(context, null));
  78. }
  79.  
  80. @Override
  81. public boolean onTouchEvent(MotionEvent e) {
  82. int x = (int)e.getX();
  83. int y = (int)e.getY();
  84. switch (e.getAction()){
  85. case MotionEvent.ACTION_DOWN:
  86. //记录当前按下的坐标
  87. xDown = x;
  88. yDown = y;
  89. //计算选中哪个Item
  90. int firstPosition = ((LinearLayoutManager)getLayoutManager()).findFirstVisibleItemPosition();
  91. Rect itemRect = new Rect();
  92.  
  93. final int count = getChildCount();
  94. for (int i=0; i<count; i++){
  95. final View child = getChildAt(i);
  96. if (child.getVisibility() == View.VISIBLE){
  97. child.getHitRect(itemRect);
  98. if (itemRect.contains(x, y)){
  99. curSelectPosition = firstPosition + i;
  100. break;
  101. }
  102. }
  103. }
  104.  
  105. if (isFirst){//第一次时,不用重置上一次的Item
  106. isFirst = false;
  107. }else {
  108. //屏幕再次接收到点击时,恢复上一次Item的状态
  109. if (mLastItemLayout != null && mMoveWidth > 0) {
  110. //将Item右移,恢复原位
  111. scrollRight(mLastItemLayout, (0 - mMoveWidth));
  112. //清空变量
  113. mHiddenWidth = 0;
  114. mMoveWidth = 0;
  115. }
  116.  
  117. }
  118.  
  119. //取到当前选中的Item,赋给mCurItemLayout,以便对其进行左移
  120. View item = getChildAt(curSelectPosition - firstPosition);
  121. if (item != null) {
  122. //获取当前选中的Item
  123. MyAdapter.ViewHolder viewHolder = (MyAdapter.ViewHolder) getChildViewHolder(item);
  124. mCurItemLayout = viewHolder.ll_item;
  125. //找到具体元素(这与实际业务相关了~~)
  126. mLlHidden = (LinearLayout)mCurItemLayout.findViewById(R.id.ll_hidden);
  127. mItemDelete = (LinearLayout)mCurItemLayout.findViewById(R.id.ll_hidden);
  128. mItemDelete.setOnClickListener(new View.OnClickListener() {
  129. @Override
  130. public void onClick(View v) {
  131. if (mRightListener != null){
  132. //删除
  133. mRightListener.onRightClick(curSelectPosition, "");
  134. }
  135. }
  136. });
  137.  
  138. //这里将删除按钮的宽度设为可以移动的距离
  139. mHiddenWidth = mLlHidden.getWidth();
  140. }
  141. break;
  142.  
  143. case MotionEvent.ACTION_MOVE:
  144. xMove = x;
  145. yMove = y;
  146. int dx = xMove - xDown;//为负时:手指向左滑动;为正时:手指向右滑动。这与Android的屏幕坐标定义有关
  147. int dy = yMove - yDown;//
  148.  
  149. //左滑
  150. if (dx < 0 && Math.abs(dx) > mTouchSlop && Math.abs(dy) < mTouchSlop){
  151. int newScrollX = Math.abs(dx);
  152. if (mMoveWidth >= mHiddenWidth){//超过了,不能再移动了
  153. newScrollX = 0;
  154. } else if (mMoveWidth + newScrollX > mHiddenWidth){//这次要超了,
  155. newScrollX = mHiddenWidth - mMoveWidth;
  156. }
  157. //左滑,每次滑动手指移动的距离
  158. scrollLeft(mCurItemLayout, newScrollX);
  159. //对移动的距离叠加
  160. mMoveWidth = mMoveWidth + newScrollX;
  161. }else if (dx > 0){//右滑
  162. //执行右滑,这里没有做跟随,瞬间恢复
  163. scrollRight(mCurItemLayout, 0 - mMoveWidth);
  164. mMoveWidth = 0;
  165. }
  166.  
  167. break;
  168. case MotionEvent.ACTION_UP://手抬起时
  169. int scrollX = mCurItemLayout.getScrollX();
  170.  
  171. if (mHiddenWidth > mMoveWidth) {
  172. int toX = (mHiddenWidth - mMoveWidth);
  173. if (scrollX > mHiddenWidth / 2) {//超过一半长度时松开,则自动滑到左侧
  174. scrollLeft(mCurItemLayout, toX);
  175. mMoveWidth = mHiddenWidth;
  176. } else {//不到一半时松开,则恢复原状
  177. scrollRight(mCurItemLayout, 0 - mMoveWidth);
  178. mMoveWidth = 0;
  179. }
  180. }
  181. mLastItemLayout = mCurItemLayout;
  182. break;
  183.  
  184. }
  185. return super.onTouchEvent(e);
  186. }
  187.  
  188. @Override
  189. public void computeScroll() {
  190. if (mScroller.computeScrollOffset()) {
  191.  
  192. Log.e(TAG, "computeScroll getCurrX ->" + mScroller.getCurrX());
  193. mCurItemLayout.scrollBy(mScroller.getCurrX(), 0);
  194. invalidate();
  195. }
  196. }
  197.  
  198. /**
  199. * 向左滑动
  200. */
  201. private void scrollLeft(View item, int scorllX){
  202. Log.e(TAG, " scroll left -> " + scorllX);
  203. item.scrollBy(scorllX, 0);
  204. }
  205.  
  206. /**
  207. * 向右滑动
  208. */
  209. private void scrollRight(View item, int scorllX){
  210. Log.e(TAG, " scroll right -> " + scorllX);
  211. item.scrollBy(scorllX, 0);
  212. }
  213.  
  214. public interface OnRightClickListener{
  215. void onRightClick(int position, String id);
  216. }
  217. }

  这样我们就实现了给RecyclerView添加左滑删除了 ,这是GitHub下载链接  See You Next Time !

Android -- RecyclerView实现左滑删除的更多相关文章

  1. Android滑动列表(拖拽,左滑删除,右滑完成)功能实现(1)

    场景: 近期做的TODO APP需要在主页添加一个功能,就是可以左滑删除,右滑完成.看了一下当前其他人做的例如仿探探式的效果,核心功能基本一样,但是和我预想的还是有少量区别,于是干脆自己重头学一遍如何 ...

  2. Android开发学习之路-PopupWindow和仿QQ左滑删除

    这周作业,要做一个类似QQ的左滑删除效果的ListView,因为不想给每个item都放一个按钮,所以决定用PopupWindow,这里记录一下 先放一下效果图: 先说明一下这里面的问题: ①没有做到像 ...

  3. [转]ANDROID仿IOS微信滑动删除_SWIPELISTVIEW左滑删除例子

    转载:http://dwtedx.sinaapp.com/itshare_290.html 本例子实现了滑动删除ListView的Itemdemo的效果.大家都知道.这种创意是来源于IOS的.左滑删除 ...

  4. Android ListView左滑删除、左滑自定义功能

    最近项目需要ListView左滑删除功能,搜集了很多资料发现了一个某一前辈写的库能很简单的实现这个功能,而且有源码,直接拿来使用了. 库名字叫做SwipeMenuListView,下面给大家演示一下使 ...

  5. 模仿QQ左滑删除

    需求: 1.左滑删除 2.向左滑动距离超过一半的时候让它自动滑开,向右滑动超过一半的时候自动隐藏 3.一次只允许滑开一个item 还有,根本不需要自定义view来实现,谨防入坑 布局: <?xm ...

  6. 类似QQ消息左滑删除的Demo

    最近在网上学到一篇类似QQ消息左滑删除的demo,完善了下代码,感觉还不错,特此分享一波: CustomSwipeListView.java 是个继承自ListView的类,里面调用了自定义View ...

  7. 仿QQ列表左滑删除

    一直想写个仿QQ通讯列表左滑删除的效果,今天终于忙里偷闲,简单一个. 大概思路是这样的: 通过 ontouchstartontouchmoveontouchend 结合css3的平移. 不多说,直接上 ...

  8. tableView左滑删除功能

    实现三个代理方法即可 -(NSString *)tableView:(UITableView *)tableView titleForDeleteConfirmationButtonForRowAtI ...

  9. wex5 实战 苹果左滑删除与长按编辑

    用了多年苹果,习惯了苹果的左滑删除与长按编辑,特别是短信什么的,很多安卓界面也采用了类似方式. 我的想法突如其来,用wex5也设计一个这样的功能,可以吗? 那句广告词,没有什么不可能. 呵呵. 一   ...

随机推荐

  1. 在U盘上安装GRUB2直接引导ISO

    本文的内容来源于 http://maxmars.net/blog/2012/10/02/boot-multiple-iso-from-usb-using-linux/ 以下所有命令都在 root 用户 ...

  2. 内置函数time

    time   import time.time() # 浮点型,给计算机看,随机 时间有三种: First: 时间戳  (time.time()) Second: 结构化时间  可以修改 Third: ...

  3. 【PyQt5-Qt Designer】QDoubleSpinBox-小数微调框

    QDoubleSpinBox-小数微调框 总体说明 大部分的总体说明和QSpinBox的差不多(详见:<PyQt5:微调框1>),这里主要把有差别的地方谈一下(三点). QDoubleSp ...

  4. word2vec训练好的词向量

    虽然早就对NLP有一丢丢接触,但是最近真正对中文文本进行处理才深深感觉到自然语言处理的难度,主要是机器与人还是有很大差异的,毕竟人和人之间都是有差异的,要不然不会讲最难研究的人嘞 ~~~~~~~~~~ ...

  5. 用A标签实现页面内容定位 点击链接跳到具体位置

    经常在维基百科等网站看到目录列表,点击链接会跳到具体的位置,小美眉一直在问是怎么做到的,其实挺简单的,用A标签实现页面内容定位就行了.实例参考微信营销理论手册的目录. 首先用A标签定义目录的链接. & ...

  6. HDFS介绍及简单操作

    目录 1.HDFS是什么? 2.HDFS设计基础与目标 3.HDFS体系结构 3.1 NameNode(NN)3.2 DataNode(DN)3.3 SecondaryNameNode(SNN)3.4 ...

  7. Auth认证模块

    1.auth简介 auth是什么? auth是django内置的用户认证系统,可以快速的实现:登录,注销,修改密码........ 2.autho的使用 1)先创建超级用户 python3 manag ...

  8. NN中的激活函数【转载】

    转自:https://blog.csdn.net/edogawachia/article/details/80043673 1.sigmoid 特点:可以解释,比如将0-1之间的取值解释成一个神经元的 ...

  9. tf中计算图 执行流程学习【转载】

    转自:https://blog.csdn.net/dcrmg/article/details/79028003 https://blog.csdn.net/qian99/article/details ...

  10. word2vec 评测 window_different

    This is a test for word2vecWed Nov 07 16:04:39 2018dir of model1: ./model/window3_ min_count2_worker ...