android - 如何在 Android 上创建半可拖动的回收 View ?

标签 android android-layout android-recyclerview

我的问题很简单,但在 Android 上实现起来非常棘手。我想知道如何只拖动标题下方的“可拖动部分”。

recyclerview table with categories

我已经按照 this tutorial 实现了拖动功能。但是我如何在拖动时区分 Header ViewHolder 和 Item ViewHolder 呢?

适配器类如下所示

public class ColorRecyclerViewAdapter extends
        RecyclerView.Adapter<ColorRecyclerViewAdapter.ItemViewHolderSelector> implements TouchCallbackHelperAdapter {

    private Context context;
    private ArrayList<String> mItems;
    private final HelperOnStartDragListener mDragStartListener;
    private ItemViewHolderSelector oldHolder;

    public ColorRecyclerViewAdapter(Context context, ArrayList<String> data, HelperOnStartDragListener dragStartListener) {
        this.context = context;
        this.mItems = data;
        this.mDragStartListener = dragStartListener;
    }

    @Override
    public void onItemDismiss(int position) {
        mItems.remove(position);
        notifyItemRemoved(position);
    }

    @Override
    public boolean onItemMove(final int fromPosition, final int toPosition) {
        if (fromPosition < toPosition) {
            for (int i = fromPosition; i < toPosition; i++) {
                Collections.swap(mItems, i, i + 1);
            }
        } else {
            for (int i = fromPosition; i > toPosition; i--) {
                Collections.swap(mItems, i, i - 1);
            }
        }
        mDragStartListener.onFinishDrag(null);
        notifyItemMoved(fromPosition, toPosition);
        return true;
    }

    public ArrayList<String> getData() {
        return mItems;
    }

    public void addItem(String item) {
        mItems.add(item);
        notifyItemInserted(mItems.size() - 1);
    }

    @Override
    public ItemViewHolderSelector onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        View view;
        view = LayoutInflater.from(context).inflate(R.layout.color_recycler_view_item_drag_mode, parent, false);
        return new ItemViewHolderSelector(view);
    }

    @Override
    public void onBindViewHolder(@NonNull final ItemViewHolderSelector holder, final int position) {
        final String item = mItems.get(position);
        holder.name.setText(item);
        holder.drag.setOnTouchListener(new View.OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                switch (event.getAction()) {
                    case MotionEvent.ACTION_DOWN:
                        mDragStartListener.onStartDrag(holder);
                        break;
                }
                return true;
            }
        });
    }

    @Override
    public int getItemCount() {
        return mItems != null ? mItems.size() : 0;
    }

    class ItemViewHolderSelector extends RecyclerView.ViewHolder implements HelperViewHolderSelector {

        private TextView name;
        private ImageView drag;
        private View itemView;

        ItemViewHolderSelector(View itemView) {
            super(itemView);
            this.itemView = itemView;
            drag = itemView.findViewById(R.id.color_recycler_view_drag);
            name = itemView.findViewById(R.id.color_recycler_view_name);
        }

        @Override
        public void onItemSelected() {
            itemView.setBackgroundColor(Color.LTGRAY);
        }

        @Override
        public void onItemClear() {
            itemView.setBackgroundColor(0);
        }
    }
}

编辑:

根据@fernandospr 的回答,我创建了一个包含最终代码的存储库。你可以 access it here

最佳答案

RecyclerView 仅显示您在 onCreateViewHolder 上创建的 View 。

根据您的职位,您需要创建不同的 View 。这意味着当位置对应于标题时,您将创建一个 HeaderViewHolder,当位置对应于一个项目时,您将创建一个 ItemViewHolder

因此,您必须将 ColorRecyclerViewAdapter 重构为:

private static final int HEADER_VIEW = 0;
private static final int ITEM_VIEW = 1;

@Override
public RecyclerView.ViewHolderSelector onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
    if (viewType == ITEM_VIEW) {
       ...
       return new ItemViewHolderSelector(view);
    } else {
       ...
       return new HeaderViewHolderSelector(view);
    }
}

@Override
public void onBindViewHolder(@NonNull final RecyclerView.ViewHolder holder, final int position) {
   if (ITEM_VIEW == holder.getItemViewType()) {
      ...
   } else {
      ...
   }
}

@Override
public int getItemCount() {
   return itemsCount + headersCount;
}

onBindViewHolder 中,您应该只允许通过设置 TouchListener 拖动项目。

此外,在 onMoveonSwiped 方法中,您应该使用位置来了解它是标题还是项目并允许或不允许执行:

@Override
public void onItemDismiss(int position) {
   if (isItemPosition(position)) { 
      ...
   }
}

@Override
public boolean onItemMove(final int fromPosition, final int toPosition) {
   if (isItemPosition(position)) { 
      ...
   }
}

isItemPosition() 是您需要编写的方法,它取决于适配器如何保存项目和 header 数据。

关于android - 如何在 Android 上创建半可拖动的回收 View ?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54096462/

相关文章:

android - 计算位图平铺高度,以便不裁剪最后一项

android - 如何与原始文件夹中的其他应用程序共享我的 mp3 文件

Android - RecyclerView 和缩放图像

android - Android 中单行的 TextView 和 EditText?

android - 使 ImageView 从方法中不可见

java - android - 为什么我的 editText 不显示?

android - LinearLayout 权重未填充父级

java - Android,项目列表未正确显示

java - android studio 尝试在空对象引用上调用虚拟方法 'void android.support.v7.widget.RecyclerView.setHasFixedSize(boolean)'

android - 如何在Android Studio中将输入数字分配为变量?