我有一个简单的RecyclerView,它使用ItemTouchHelper来滑动和拖动项目。一切正常,直到我需要以不同方式设置第一个和最后一个项目的样式,所以我需要覆盖适配器中的getItemViewType(int p)函数。在我这样做之后,拖动功能停止流畅,并且在向上/向下移动一个位置后总是丢弃项目。这是我的RecyclerView适配器:
public class CurrentStopsAdapter extends RecyclerView.Adapter<CurrentStopsAdapter.ViewHolder> {
private RemoveFromTripListener removeFromTripListener = null;
private SwapPlacesTripListener swapPlacesTripListener = null;
private Context context;
private List<PlaceLink> stops;
public CurrentStopsAdapter(Context context, List<PlaceLink> stops,
RemoveFromTripListener removeListener, SwapPlacesTripListener swapListener) {
this.context = context;
this.stops = stops;
removeFromTripListener = removeListener;
swapPlacesTripListener = swapListener;
}
public interface RemoveFromTripListener {
void onRemoveTripButtonClick(String id);
}
public interface SwapPlacesTripListener {
void onSwapPlacesButtonClick(int first, int second);
}
@Override
public int getItemViewType(int position) {
if (position == (getItemCount()-1)) {
return -1;
}
return position;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
switch (viewType) {
case -1:
return new ViewHolder(LayoutInflater.from(context).inflate(R.layout.list_item_destination, parent, false));
case 0:
return new ViewHolder(LayoutInflater.from(context).inflate(R.layout.list_item_origin, parent, false));
default:
return new ViewHolder(LayoutInflater.from(context).inflate(R.layout.list_item_stop, parent, false));
}
}
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
holder.bindStop(stops.get(position));
}
@Override
public int getItemCount() {
return stops.size();
}
public void remove(int position) {
if (removeFromTripListener != null) {
removeFromTripListener.onRemoveTripButtonClick(stops.get(position).getId());
}
notifyItemRemoved(position);
}
public void swap(int firstPosition, int secondPosition) {
if (swapPlacesTripListener != null) {
swapPlacesTripListener.onSwapPlacesButtonClick(firstPosition, secondPosition);
}
notifyItemMoved(firstPosition, secondPosition);
}
public class ViewHolder extends RecyclerView.ViewHolder {
public final TextView placeTitle;
public ViewHolder(View view){
super(view);
placeTitle = (TextView) view.findViewById(R.id.stops_list_item_title);
}
public void bindStop(PlaceLink place){
this.placeTitle.setText(place.getTitle());
}
}
}
和我的ItemTouchHelper:
public class TripTouchHelper extends ItemTouchHelper.SimpleCallback {
private CurrentStopsAdapter currentStopsAdapter;
public TripTouchHelper(CurrentStopsAdapter currentStopsAdapter){
super(ItemTouchHelper.UP | ItemTouchHelper.DOWN, ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT);
this.currentStopsAdapter = currentStopsAdapter;
}
@Override
public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {
currentStopsAdapter.swap(viewHolder.getAdapterPosition(), target.getAdapterPosition());
return true;
}
@Override
public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {
currentStopsAdapter.remove(viewHolder.getAdapterPosition());
}
}
如果我跳过getItemViewType上的覆盖并对所有项目使用相同的布局,一切正常但我非常希望找到解决此问题的方法。任何帮助将非常感激:)
答案 0 :(得分:2)
在recyclerview适配器中删除getItemViewType(int position)。我遇到了同样的问题,这很有用。
答案 1 :(得分:1)
适配器认为您移动的ViewHolder
被认为是不同类型,因为getItemViewType()
返回了唯一的类型。所以它应该重新渲染。
例如,您可以按照我的程序返回类型:
return 0
表示r.layout.foo
return 1
然后,当您希望按类型进行布局时,您可以编写自己的方法/对来检查布局。如果您仍需要职位,则可以在创建ViewHolder
后获得该职位:AdapterPosition
。
如果你需要OnCreateViewHolder()
中的位置,你就会出错,可能会失去ViewHolder
的目的。考虑在操作数据时执行该工作,或者onBindViewHolder()
如果不是那么昂贵。
答案 2 :(得分:0)
我认为您应该将“类型”作为相关项目的int返回,而不是位置。这意味着,对于页眉/页脚类型,您可以返回一个值(例如:1),而对于所有其他类型,您可以返回另一个值(例如:2)。返回位置通过您的类型将每个移动视为不同的视图类型,因为位置将是唯一的,我认为这会阻止拖动。