当前位置:网站首页>RecyclerView高级使用(二)-垂直拖拽排序的简单实现

RecyclerView高级使用(二)-垂直拖拽排序的简单实现

2022-04-23 14:06:00 森之千手

先看看要实现的效果图:
![简单的垂直拖拽排序](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返回的是DOWNUP
2、在onMove中,pos由于是从零开始计算的,所以实际的刷新条目个数要+1
3、在onMove中,关于数据的交换,采用了系统Collections的swap交换,更为高效
4、isLongPressDragEnabled表示是否是长按后才启用拖拽,这里需要返回true,表示需要

源码下载地址:github

版权声明
本文为[森之千手]所创,转载请带上原文链接,感谢
https://blog.csdn.net/cjs1534717040/article/details/115300429