在安卓中,有许多关于如何使用RecyclerView实现“drag & drop”与swipe-to-dismiss”的教程,库和例子。即使现在已经有了新的,更优的实现方式,大多数仍然是使用老旧的View.OnDragListener以及Roman
Nurik在SwipeToDismiss中所使用的方法。很少有人使用新的api,反而要么经常依赖于GestureDetectors和onInterceptTouchEvent,要么实现方式很复杂。实际上,在RecyclerView上添加拖动特性有一个非常简单的方法。这个方法只需要一个类,并且它也是Android
兼容包的一部分,它就是:
ItemTouchHelper
ItemTouchHelper是一个强大的工具,它处理好了关于在RecyclerView上添加拖动排序与滑动删除的所有事情。它是RecyclerView.ItemDecoration的子类,也就是说它可以轻易的添加到几乎所有的LayoutManager和Adapter中。它还可以和现有的item动画一起工作,提供受类型限制的拖放动画等等。
使用 ItemTouchHelper 和 ItemTouchHelper.Callback
要使用ItemTouchHelper,你需要创建一个ItemTouchHelper.Callback。这个接口可以让你监听“move”与 “swipe”事件。这里还是控制view被选中的状态以及重写默认动画的地方。如果你只是想要一个基本的实现,有一个帮助类可以使用:SimpleCallback,但是为了了解其工作机制,我们还是自己实现。
public class SimpleItemTouchHelperCallback extends ItemTouchHelper.Callback {
public SimpleItemTouchHelperCallback(ItemTouchHelperAdapter adapter) {
}
@Override
public boolean isLongPressDragEnabled() {
return true;
}
@Override
public boolean isItemViewSwipeEnabled() {
return true;
}
@Override
public int getMovementFlags(RecyclerView recyclerView, ViewHolder viewHolder) {//重写getMoivementFlags()方法来指
int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN; //定可以支持的拖动和滑动的方向。使用
int swipeFlags = ItemTouchHelper.START | ItemTouchHelper.END; //helperItemToooouchHelper.makeMovementFlages
return makeMovementFlags(dragFlags, swipeFlags); //(int,int)来构造返回的flag。上下为drag,左右为swipe
}
@Override
public boolean onMove(RecyclerView recyclerView, ViewHolder viewHolder,//拖动事件响应,重新排序
ViewHolder target) {
int fromPosition = viewHolder.getAdapterPosition)();
int targetPosition = target.getAdapterPosition();
Collections.swap(mItems,fromPosion,targetPosition);
notifyItemMove(fromPosition,targetPosition);
return true;
}
@Override
public void onSwiped(ViewHolder viewHolder, int direction) {//左右滑动事件响应,删除item
int position = viewHOlder.getAdapterPosition();
mItems.remove(position);
notifyItemRemoved(position);
}
}
Callback准备好了之后,我们就可以创建我们的ItemTouchHelper并调用attachToRecyclerView(RecyclerView) 了
ItemTouchHelper.Callback callback =
new SimpleItemTouchHelperCallback(adapter);
ItemTouchHelper touchHelper = new ItemTouchHelper(callback);
touchHelper.attachToRecyclerView(recyclerView);
运行之后,你可以看到如下的效果: