RecyclerView高级使用(二)-垂直拖拽排序的简单实现
程序员文章站
2022-05-04 20:02:18
...
先看看要实现的效果图:
![简单的垂直拖拽排序](https://img-blog.csdnimg.cn/20210329142737702.gif#pic_center
效果比较简单,就是一个垂直列表,然后可以拖动其子条目进行排序。
因此采用的方式还是RecyclerView+ItemTouchHelper,关于ItemTouchHelper的相关说明及使用还可以参考RecyclerView高级使用(一)-侧滑删除的简单实现和RecyclerView细节研究-RecyclerView点击错位问题的探讨与修复。这里我们需要自己手写一个类VerticalDragSortHelperCallBack去继承ItemTouchHelper.Callback,我们需要实现下面的方法:
@Override
public int getMovementFlags(@NonNull RecyclerView recyclerView, @NonNull RecyclerView.ViewHolder viewHolder) {
int dragFlag = ItemTouchHelper.UP | ItemTouchHelper.DOWN;
return makeMovementFlags(dragFlag, ItemTouchHelper.ACTION_STATE_IDLE);
}
@Override
public boolean onMove(@NonNull RecyclerView recyclerView, @NonNull RecyclerView.ViewHolder viewHolder, @NonNull RecyclerView.ViewHolder target) {
int fromPos = viewHolder.getAdapterPosition();
int toPos = target.getAdapterPosition();
Log.d("onMove", fromPos + "--->" + toPos);
//1、从数据源上交换条目的位置
Collections.swap(recyclerItemList, fromPos, toPos);
//2、从视图上通知刷新视图的改变
recyclerView.getAdapter().notifyItemMoved(fromPos, toPos);
//3、更正实际点击的position,防止点击时,position错乱
int startPos = Math.min(fromPos, toPos);
int itemCount = Math.abs(fromPos - toPos) + 1;
recyclerView.getAdapter().notifyItemRangeChanged(startPos, itemCount);
//4、PS:2和3的合体步骤可以用4替代,但是就没有了替换动画了,而且刷新效率比上面低,是全局刷新
// recyclerView.getAdapter().notifyDataSetChanged();
//请注意该返回值:只有在返回true的时候,才会走onMoved方法
return true;
}
@Override
public void onMoved(@NonNull RecyclerView recyclerView, @NonNull RecyclerView.ViewHolder viewHolder, int fromPos, @NonNull RecyclerView.ViewHolder target, int toPos, int x, int y) {
super.onMoved(recyclerView, viewHolder, fromPos, target, toPos, x, y);
Log.d("onMoved", "第一个:" + recyclerItemList.get(0).toString() + " " + fromPos + "--->" + toPos);
Log.d("onMoved", "第二个:" + recyclerItemList.get(1).toString() + " " + fromPos + "--->" + toPos);
if (onDragListener != null) {
onDragListener.onItemMoved(viewHolder, target, fromPos, toPos);
}
}
@Override
public boolean isLongPressDragEnabled() {
return true;
}
需要注意的是:
1、因为垂直方向拖拽,所以getMovemonetFlags返回的是DOWN和UP
2、在onMove中,pos由于是从零开始计算的,所以实际的刷新条目个数要+1
3、在onMove中,关于数据的交换,采用了系统Collections的swap交换,更为高效
4、isLongPressDragEnabled表示是否是长按后才启用拖拽,这里需要返回true,表示需要
源码下载地址:github