原生就自带有可拖动item的工具:ItemTouchHelper
看下效果:
可拖动RecyclerView预览效果
接下来我们看如何使用。
1、自定义ItemTouchHelper的callback,用来限制是否可以拖动,以及拖动之后的位置更新:
其中判断条件中的item.isMovable这边是记录该item是否可以拖动,也可以换成其他判断条件比如根据位置判断等。
private static class MyItemTouchHelperCallback extends ItemTouchHelper.Callback {private final ItemAdapter itemAdapter;public MyItemTouchHelperCallback(ItemAdapter itemAdapter) {this.itemAdapter= itemAdapter;}@Overridepublic int getMovementFlags(@NonNull RecyclerView recyclerView,@NonNull RecyclerView.ViewHolder viewHolder) {int layoutPosition = viewHolder.getLayoutPosition();ItemInfo item = ItemAdapter.getItem(layoutPosition);if (!item.isMovable()) {//不可拖动return makeMovementFlags(0, 0);}//这里表示可以拖动的方向,比如如果不给往上方拖动则去掉ItemTouchHelper.UPfinal int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN| ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;return makeFlag(ItemTouchHelper.ACTION_STATE_DRAG, dragFlags);}@Overridepublic boolean onMove(@NonNull RecyclerView recyclerView,@NonNull RecyclerView.ViewHolder viewHolderSource,@NonNull RecyclerView.ViewHolder viewHolderTarget) {int layoutPosition = viewHolderTarget.getLayoutPosition();ItemInfo item = itemAdapter.getItem(layoutPosition);if (!item.isMovable()) {//不可拖动到这里return false;}itemAdapter.onMove(viewHolderSource.getAdapterPosition(),viewHolderTarget.getAdapterPosition());return true;}@Overridepublic void onSwiped(@NonNull RecyclerView.ViewHolder viewHolder, int i) {}}
2、在适配器提供获取item信息和移动之后更新数据的方法给到这个callback使用:
public ItemInfo getItem(int position) {if (position >= 0 && position < mList.size()) {return mList.get(position);}return null;}public void onMove(int sourcePosition, int targetPosition) {ItemInfo item = mList.get(sourcePosition);mList.remove(sourcePosition);mList.add(targetPosition, item);notifyItemMoved(sourcePosition, targetPosition);}
3、使用这个callback类:
ItemTouchHelper itemTouchHelper = new ItemTouchHelper(new MyItemTouchHelperCallback(adapter));itemTouchHelper.attachToRecyclerView(recyclerView);
就可以了,简简单单,轻轻松松。
另外,需要留意一点,如果你是在onBindViewHolder中有设置点击事件的,在onClickListener里面不要直接使用onBindViewHolder方法传进来的position,因为在item位置移动之后,这个position是不会变的,注册事件监听的时候这个值就跟你的点击事件绑死了,所以需要动态获取这个position:
holder.view.setOnClickListener(v -> {int currentPosition = holder.getAdapterPosition();ItemInfo info = getItem(currentPosition);if (listener != null) {listener.onItemClick(currentPosition, info);}});