原生就自带有可拖动item的工具:ItemTouchHelper
看下效果:
接下来我们看如何使用。
1、自定义ItemTouchHelper的callback,用来限制是否可以拖动,以及拖动之后的位置更新:
其中判断条件中的item.isMovable这边是记录该item是否可以拖动,也可以换成其他判断条件比如根据位置判断等。
private static class MyItemTouchHelperCallback extends ItemTouchHelper.Callback { private final ItemAdapter itemAdapter; public MyItemTouchHelperCallback(ItemAdapter itemAdapter) { this.itemAdapter= itemAdapter; } @Override public 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.UP final int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN | ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT; return makeFlag(ItemTouchHelper.ACTION_STATE_DRAG, dragFlags); } @Override public 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; } @Override public 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); } });