当前位置:   android > 正文

Android 实现菜单拖拽排序_android recyclerview 拖动排序

android recyclerview 拖动排序

效果图

简介

本文主角是ItemTouchHelper。

它是RecyclerView对于item交互处理的一个「辅助类」,主要用于拖拽以及滑动处理。

以接口实现的方式,达到配置简单、逻辑解耦、职责分明的效果,并且支持所有的布局方式。

功能拆解

功能实现

4.1、实现接口

自定义一个类,实现ItemTouchHelper.Callback接口,然后在实现方法中根据需求简单配置即可。

  1. class DragCallBack(adapter: DragAdapter, data: MutableList<String>) : ItemTouchHelper.Callback() {
  2. }

ItemTouchHelper.Callback必须实现的3个方法:

  • getMovementFlags

  • onMove

  • onSwiped

其他方法还有onSelectedChanged、clearView等。

4.1.1、getMovementFlags

用于创建交互方式,交互方式分为两种:

1. 拖拽,网格布局支持上下左右,列表只支持上下(LEFT、UP、RIGHT、DOWN)。

2. 滑动,只支持前后(START、END)。

最后,通过makeMovementFlags把结果返回回去,makeMovementFlags接收两个参数,dragFlags和swipeFlags,即上面拖拽和滑动组合的标志位

  1. override fun getMovementFlags(recyclerView: RecyclerView, viewHolder: RecyclerView.ViewHolder): Int {
  2. var dragFlags = 0
  3. var swipeFlags = 0
  4. when (recyclerView.layoutManager) {
  5. is GridLayoutManager -> {
  6. // 网格布局
  7. dragFlags = ItemTouchHelper.LEFT or ItemTouchHelper.UP or ItemTouchHelper.RIGHT or ItemTouchHelper.DOWN
  8. return makeMovementFlags(dragFlags, swipeFlags)
  9. }
  10. is LinearLayoutManager -> {
  11. // 线性布局
  12. dragFlags = ItemTouchHelper.UP or ItemTouchHelper.DOWN
  13. swipeFlags = ItemTouchHelper.START or ItemTouchHelper.END
  14. return makeMovementFlags(dragFlags, swipeFlags)
  15. }
  16. else -> {
  17. // 其他情况可自行处理
  18. return 0
  19. }
  20. }
  21. }

4.1.2、onMove

拖拽时回调,这里我们主要对起始位置和目标位置的item做一个数据交换,然后刷新视图显示。

  1. override fun onMove(recyclerView: RecyclerView, viewHolder: RecyclerView.ViewHolder, target: RecyclerView.ViewHolder): Boolean {
  2. // 起始位置
  3. val fromPosition = viewHolder.adapterPosition
  4. // 结束位置
  5. val toPosition = target.adapterPosition
  6. // 固定位置
  7. if (fromPosition == mAdapter.fixedPosition || toPosition == mAdapter.fixedPosition) {
  8. return false
  9. }
  10. // 根据滑动方向 交换数据
  11. if (fromPosition < toPosition) {
  12. // 含头不含尾
  13. for (index in fromPosition until toPosition) {
  14. Collections.swap(mData, index, index + 1)
  15. }
  16. } else {
  17. // 含头不含尾
  18. for (index in fromPosition downTo toPosition + 1) {
  19. Collections.swap(mData, index, index - 1)
  20. }
  21. }
  22. // 刷新布局
  23. mAdapter.notifyItemMoved(fromPosition, toPosition)
  24. return true
  25. }

4.1.3、onSwiped

滑动时回调,这个回调方法里主要是做数据和视图的更新操作。

  1. override fun onSwiped(viewHolder: RecyclerView.ViewHolder, direction: Int) {
  2. if (direction == ItemTouchHelper.START) {
  3. Log.i(TAG, "START--->向左滑")
  4. } else {
  5. Log.i(TAG, "END--->向右滑")
  6. }
  7. val position = viewHolder.adapterPosition
  8. mData.removeAt(position)
  9. mAdapter.notifyItemRemoved(position)
  10. }

4.2、绑定RecyclerView

上面接口实现部分我们已经简单写好了,逻辑也挺简单,总共不超过100行代码。

接下来就是把这个辅助类绑定到RecyclerView。

RecyclerView显示的实现就是基础的样式,就不展开了,可以查看源码。

  1. val dragCallBack = DragCallBack(mAdapter, list)
  2. val itemTouchHelper = ItemTouchHelper(dragCallBack)
  3. itemTouchHelper.attachToRecyclerView(mBinding.recycleView)

绑定只需要调用attachToRecyclerView就好了。

至此,简单的效果就已经实现了。下面开始优化和进阶的部分。

4.3、设置分割线

RecyclerView网格布局实现等分,我们一般先是自定义ItemDecoration,然后调用addItemDecoration来实现的。

但是我在实现效果的时候遇到一个问题,因为我加了布局切换的功能,在每次切换的时候,针对不同的布局分别设置layoutManager和ItemDecoration,这就导致随着切换次数的增加,item的间隔就越大。

addItemDecoration,顾名思义是添加,通过查看源码发现RecyclerView内部是有一个ArrayList来维护的,所以当我们重复调用addItemDecoration方法时,分割线是以递增的方式在增加的,并且在绘制的时候会从集合中遍历所有的分割线绘制。

部分源码:

  1. @Override
  2. public void draw(Canvas c) {
  3. super.draw(c);
  4. final int count = mItemDecorations.size();
  5. for (int i = 0; i < count; i++) {
  6. mItemDecorations.get(i).onDrawOver(c, this, mState);
  7. }
  8. //...
  9. }

既然知道了问题所在,也大概想到了3种解决办法:

1. 调用addItemDecoration前,先调用removeItemDecoration方法remove掉之前所有的分割线。

2. 调用addItemDecoration(@NonNull ItemDecoration decor, int index),通过index来维护。

3. add时通过一个标示来判断,添加过就不添加了。

好像可行,实际上并不太行...因为始终都有两个分割线实例。

我们再来梳理一下:

  • 两种不同的布局

  • 都有分割线

  • 分割线只需设置一次

我想到另外一个办法,不对RecyclerView做处理了,既然两种布局都有分割线,是不是可以把分割线合二为一了,然后根据LayoutManager去绘制不同的分割线?

理论上是可行的,事实上也确实可以...

自定义分割线:

  1. class GridSpaceItemDecoration(private val spanCount: Int, private val spacing: Int = 20, private var includeEdge: Boolean = false) :
  2. RecyclerView.ItemDecoration() {
  3. override fun getItemOffsets(outRect: Rect, view: View, recyclerView: RecyclerView, state: RecyclerView.State) {
  4. recyclerView.layoutManager?.let {
  5. when (recyclerView.layoutManager) {
  6. is GridLayoutManager -> {
  7. val position = recyclerView.getChildAdapterPosition(view) // 获取item在adapter中的位置
  8. val column = position % spanCount // item所在的列
  9. if (includeEdge) {
  10. outRect.left = spacing - column * spacing / spanCount
  11. outRect.right = (column + 1) * spacing / spanCount
  12. if (position < spanCount) {
  13. outRect.top = spacing
  14. }
  15. outRect.bottom = spacing
  16. } else {
  17. outRect.left = column * spacing / spanCount
  18. outRect.right = spacing - (column + 1) * spacing / spanCount
  19. if (position >= spanCount) {
  20. outRect.top = spanCount
  21. }
  22. outRect.bottom = spacing
  23. }
  24. }
  25. is LinearLayoutManager -> {
  26. outRect.top = spanCount
  27. outRect.bottom = spacing
  28. }
  29. }
  30. }
  31. }
  32. }

4.4、选中放大/背景变色

为了提升用户体验,可以在拖拽的时候告诉用户当前拖拽的是哪个item,比如选中的item放大、背景高亮等。

  • 网格布局,选中变大。

  • 列表布局,背景变色。

这里用到ItemTouchHelper.Callback中的两个方法,onSelectedChanged和clearView,我们需要在选中时改变视图显示,结束时再恢复。

4.4.1、onSelectedChanged

拖拽或滑动 发生改变时回调,这时我们可以修改item的视图。

  1. override fun onSelectedChanged(viewHolder: RecyclerView.ViewHolder?, actionState: Int) {
  2. if (actionState != ItemTouchHelper.ACTION_STATE_IDLE) {
  3. viewHolder?.let {
  4. // 因为拿不到recyclerView,无法通过recyclerView.layoutManager来判断是什么布局,所以用item的宽度来判断
  5. // itemView.width > 500 用这个来判断是否是线性布局,实际取值自己看情况
  6. if (it.itemView.width > 500) {
  7. // 线性布局 设置背景颜色
  8. val drawable = it.itemView.background as GradientDrawable
  9. drawable.color = ContextCompat.getColorStateList(it.itemView.context, R.color.greenDark)
  10. } else {
  11. // 网格布局 设置选中放大
  12. ViewCompat.animate(it.itemView).setDuration(200).scaleX(1.3F).scaleY(1.3F).start()
  13. }
  14. }
  15. }
  16. super.onSelectedChanged(viewHolder, actionState)
  17. }

actionState:

  • ACTION_STATE_IDLE 空闲状态。

  • ACTION_STATE_SWIPE 滑动状态。

  • ACTION_STATE_DRAG 拖拽状态。

4.4.2、clearView

拖拽或滑动 结束时回调,这时我们要把改变后的item视图恢复到初始状态。

  1. override fun clearView(recyclerView: RecyclerView, viewHolder: RecyclerView.ViewHolder) {
  2. // 恢复显示
  3. // 这里不能用if判断,因为GridLayoutManager是LinearLayoutManager的子类,改用when,类型推导有区别
  4. when (recyclerView.layoutManager) {
  5. is GridLayoutManager -> {
  6. // 网格布局 设置选中大小
  7. ViewCompat.animate(viewHolder.itemView).setDuration(200).scaleX(1F).scaleY(1F).start()
  8. }
  9. is LinearLayoutManager -> {
  10. // 线性布局 设置背景颜色
  11. val drawable = viewHolder.itemView.background as GradientDrawable
  12. drawable.color = ContextCompat.getColorStateList(viewHolder.itemView.context, R.color.greenPrimary)
  13. }
  14. }
  15. super.clearView(recyclerView, viewHolder)
  16. }

4.5、固定位置

在实际需求中,交互可能要求我们第一个菜单不可以变更顺序,只能固定,比如效果中的第一个菜单「推荐」固定在首位这种情况。

4.5.1、修改adapter

定义一个固定值,并设置不同的背景色和其他菜单区分开。

  1. class DragAdapter(private val mContext: Context, private val mList: List<String>) : RecyclerView.Adapter<DragAdapter.ViewHolder>() {
  2. val fixedPosition = 0 // 固定菜单
  3. override fun onBindViewHolder(holder: ViewHolder, position: Int) {
  4. holder.mItemTextView.text = mList[position]
  5. // 第一个固定菜单
  6. val drawable = holder.mItemTextView.background as GradientDrawable
  7. if (holder.adapterPosition == 0) {
  8. drawable.color = ContextCompat.getColorStateList(mContext, R.color.greenAccent)
  9. }else{
  10. drawable.color = ContextCompat.getColorStateList(mContext, R.color.greenPrimary)
  11. }
  12. }
  13. //...
  14. }

4.5.1、修改onMove回调

在onMove方法中判断,只要是固定位置就直接返回false。

  1. class DragCallBack(adapter: DragAdapter, data: MutableList<String>) : ItemTouchHelper.Callback() {
  2. /**
  3. * 拖动时回调
  4. */
  5. override fun onMove(recyclerView: RecyclerView, viewHolder: RecyclerView.ViewHolder, target: RecyclerView.ViewHolder): Boolean {
  6. // 起始位置
  7. val fromPosition = viewHolder.adapterPosition
  8. // 结束位置
  9. val toPosition = target.adapterPosition
  10. // 固定位置
  11. if (fromPosition == mAdapter.fixedPosition || toPosition == mAdapter.fixedPosition) {
  12. return false
  13. }
  14. // ...
  15. return true
  16. }
  17. }

虽然第一个菜单无法交换位置了,但是它还是可以拖拽的。

效果实现了吗,好像也实现了,可是又好像哪里不对,就好像填写完表单点击提交时你告诉我格式不正确一样,你不能一开始就告诉我吗?

为了进一步提升用户体验,可以让固定位置不可以拖拽吗?

可以,ItemTouchHelper.Callback中有两个方法:

1. isLongPressDragEnabled 是否可以长按拖拽。

2. isItemViewSwipeEnabled 是否可以滑动。

这俩方法默认都是true,所以即使不能交换位置,但默认也是支持操作的。

4.5.3、重写isLongPressDragEnabled

以拖拽举例,我们需要重写isLongPressDragEnabled方法把它禁掉,然后在非固定位置的时候去手动开启。

  1. override fun isLongPressDragEnabled(): Boolean {
  2. //return super.isLongPressDragEnabled()
  3. return false
  4. }

禁掉之后什么时候再触发呢?

因为我们现在的交互是长按进入编辑,那就需要在长按事件中再调用startDrag手动开启。

  1. mAdapter.setOnItemClickListener(object : DragAdapter.OnItemClickListener {
  2. //...
  3. override fun onItemLongClick(holder: DragAdapter.ViewHolder) {
  4. if (holder.adapterPosition != mAdapter.fixedPosition) {
  5. itemTouchHelper.startDrag(holder)
  6. }
  7. }
  8. })

ok,这样就完美实现了。

4.6、其他

4.6.1、position

因为有拖拽操作,下标其实是变化的,在做相应的操作时,要取实时位置。

holder.adapterPosition

4.6.2、重置

不管是拖拽还是滑动,其实本质都是对Adapter内已填充的数据进行操作,实时数据通过Adapter获取即可。

如果想要实现重置功能,直接拿最开始的原始数据重新塞给Adapter即可。

源码探索

看源码时,找对一个切入点,往往能达到事半功倍的效果。

这里就从绑定RecyclerView开始吧。

  1. val dragCallBack = DragCallBack(mAdapter, list)
  2. val itemTouchHelper = ItemTouchHelper(dragCallBack)
  3. itemTouchHelper.attachToRecyclerView(mBinding.recycleView)

实例化ItemTouchHelper,然后调用其attachToRecyclerView方法绑定到RecyclerView。

5.1、attachToRecyclerView

  1. public void attachToRecyclerView(@Nullable RecyclerView recyclerView) {
  2. if (mRecyclerView == recyclerView) {
  3. return; // nothing to do
  4. }
  5. if (mRecyclerView != null) {
  6. destroyCallbacks();
  7. }
  8. mRecyclerView = recyclerView;
  9. if (recyclerView != null) {
  10. final Resources resources = recyclerView.getResources();
  11. mSwipeEscapeVelocity = resources.getDimension(R.dimen.item_touch_helper_swipe_escape_velocity);
  12. mMaxSwipeVelocity = resources.getDimension(R.dimen.item_touch_helper_swipe_escape_max_velocity);
  13. setupCallbacks();
  14. }
  15. }

这段代码其实有点意思的,解读一下:

1. 第一个if判断,避免重复操作,直接return。

2. 第二个if判断,调用了destroyCallbacks,在destroyCallbacks里面做了一些移除和回收操作,说明只能绑定到一个RecyclerView;同时,注意这里判断的主体是mRecyclerView,不是我们传进来的recyclerView,而且我们传进来的recyclerView是支持Nullable的,所以我们可以传个空值走到destroyCallbacks里来做解绑操作。

3. 第三个if判断,当我们传的recyclerView不为空时,调用setupCallbacks。

5.2、setupCallbacks

  1. private void setupCallbacks() {
  2. ViewConfiguration vc = ViewConfiguration.get(mRecyclerView.getContext());
  3. mSlop = vc.getScaledTouchSlop();
  4. mRecyclerView.addItemDecoration(this);
  5. mRecyclerView.addOnItemTouchListener(mOnItemTouchListener);
  6. mRecyclerView.addOnChildAttachStateChangeListener(this);
  7. startGestureDetection();
  8. }

这个方法里已经大概可以看出内部实现原理了。

两个关键点:

  • addOnItemTouchListener

  • startGestureDetection

通过触摸和手势识别来处理交互显示。

5.3、mOnItemTouchListener

  1. private final OnItemTouchListener mOnItemTouchListener = new OnItemTouchListener() {
  2. @Override
  3. public boolean onInterceptTouchEvent(@NonNull RecyclerView recyclerView, @NonNull MotionEvent event) {
  4. mGestureDetector.onTouchEvent(event);
  5. if (action == MotionEvent.ACTION_DOWN) {
  6. //...
  7. if (mSelected == null) {
  8. if (animation != null) {
  9. //...
  10. select(animation.mViewHolder, animation.mActionState);
  11. }
  12. }
  13. } else if (action == MotionEvent.ACTION_CANCEL || action == MotionEvent.ACTION_UP) {
  14. select(null, ACTION_STATE_IDLE);
  15. } else if (mActivePointerId != ACTIVE_POINTER_ID_NONE) {
  16. //...
  17. if (index >= 0) {
  18. checkSelectForSwipe(action, event, index);
  19. }
  20. }
  21. return mSelected != null;
  22. }
  23. @Override
  24. public void onTouchEvent(@NonNull RecyclerView recyclerView, @NonNull MotionEvent event) {
  25. mGestureDetector.onTouchEvent(event);
  26. //...
  27. if (activePointerIndex >= 0) {
  28. checkSelectForSwipe(action, event, activePointerIndex);
  29. }
  30. switch (action) {
  31. case MotionEvent.ACTION_MOVE: {
  32. if (activePointerIndex >= 0) {
  33. moveIfNecessary(viewHolder);
  34. }
  35. break;
  36. }
  37. //...
  38. }
  39. }
  40. @Override
  41. public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {
  42. select(null, ACTION_STATE_IDLE);
  43. }
  44. };

这段代码删减之后还是有点多,不过没关系,提炼一下,核心通过判断MotionEvent调用了几个方法:

  • select

  • checkSelectForSwipe

  • moveIfNecessary

5.3.1、select

  1. void select(@Nullable ViewHolder selected, int actionState) {
  2. if (selected == mSelected && actionState == mActionState) {
  3. return;
  4. }
  5. //...
  6. if (mSelected != null) {
  7. if (prevSelected.itemView.getParent() != null) {
  8. final float targetTranslateX, targetTranslateY;
  9. switch (swipeDir) {
  10. case LEFT:
  11. case RIGHT:
  12. case START:
  13. case END:
  14. targetTranslateY = 0;
  15. targetTranslateX = Math.signum(mDx) * mRecyclerView.getWidth();
  16. break;
  17. //...
  18. }
  19. //...
  20. } else {
  21. removeChildDrawingOrderCallbackIfNecessary(prevSelected.itemView);
  22. mCallback.clearView(mRecyclerView, prevSelected);
  23. }
  24. }
  25. //...
  26. mCallback.onSelectedChanged(mSelected, mActionState);
  27. mRecyclerView.invalidate();
  28. }

这里面主要是在拖拽或滑动时对translateX/Y的计算和处理,然后通过mCallback.clearView和mCallback.onSelectedChanged回调给我们,最后调用invalidate()实时刷新。

5.3.2、checkSelectForSwipe

  1. void checkSelectForSwipe(int action, MotionEvent motionEvent, int pointerIndex) {
  2. //...
  3. if (absDx < mSlop && absDy < mSlop) {
  4. return;
  5. }
  6. if (absDx > absDy) {
  7. if (dx < 0 && (swipeFlags & LEFT) == 0) {
  8. return;
  9. }
  10. if (dx > 0 && (swipeFlags & RIGHT) == 0) {
  11. return;
  12. }
  13. } else {
  14. if (dy < 0 && (swipeFlags & UP) == 0) {
  15. return;
  16. }
  17. if (dy > 0 && (swipeFlags & DOWN) == 0) {
  18. return;
  19. }
  20. }
  21. select(vh, ACTION_STATE_SWIPE);
  22. }

这里是滑动处理的check,最后也是收敛到select()方法统一处理。

5.3.3、moveIfNecessary

  1. void moveIfNecessary(ViewHolder viewHolder) {
  2. if (mRecyclerView.isLayoutRequested()) {
  3. return;
  4. }
  5. if (mActionState != ACTION_STATE_DRAG) {
  6. return;
  7. }
  8. //...
  9. if (mCallback.onMove(mRecyclerView, viewHolder, target)) {
  10. // keep target visible
  11. mCallback.onMoved(mRecyclerView, viewHolder, fromPosition,
  12. target, toPosition, x, y);
  13. }
  14. }

这里检查拖拽时是否需要交换item,通过mCallback.onMoved回调给我们。

5.4、startGestureDetection

  1. private void startGestureDetection() {
  2. mItemTouchHelperGestureListener = new ItemTouchHelperGestureListener();
  3. mGestureDetector = new GestureDetectorCompat(mRecyclerView.getContext(),
  4. mItemTouchHelperGestureListener);
  5. }

5.4.1、ItemTouchHelperGestureListener

  1. private class ItemTouchHelperGestureListener extends GestureDetector.SimpleOnGestureListener {
  2. //...
  3. @Override
  4. public void onLongPress(MotionEvent e) {
  5. //...
  6. View child = findChildView(e);
  7. if (child != null) {
  8. ViewHolder vh = mRecyclerView.getChildViewHolder(child);
  9. if (vh != null) {
  10. //...
  11. if (pointerId == mActivePointerId) {
  12. //...
  13. if (mCallback.isLongPressDragEnabled()) {
  14. select(vh, ACTION_STATE_DRAG);
  15. }
  16. }
  17. }
  18. }
  19. }
  20. }

这里主要是对长按事件的处理,最后也是收敛到select()方法统一处理。

5.5、源码小结

1. 绑定RecyclerView。

2. 注册触摸手势监听。

3. 根据手势,先是内部处理各种校验、位置计算、动画处理、刷新等,然后回调给ItemTouchHelper.Callback。

事儿大概就是这么个事儿,主要工作都是源码帮我们做了,我们只需要在回调里根据结果处理业务逻辑即可。

源码地址

https://github.com/yechaoa/MaterialDesign

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/blog/android/detail/33725
推荐阅读
相关标签
  

闽ICP备14008679号