Android RecyclerView добавление и удаление элементов
у меня есть RecyclerView с текстовым полем TextView и перекрестной кнопкой ImageView. У меня есть кнопка за пределами recyclerview, которая делает крестовую кнопку ImageView видимой / исчезнувшей.
Я ищу, чтобы удалить элемент из recylerview, когда что элементы крест кнопка ImageView нажата.
мой адаптер:
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> implements View.OnClickListener, View.OnLongClickListener {
private ArrayList<String> mDataset;
private static Context sContext;
public MyAdapter(Context context, ArrayList<String> myDataset) {
mDataset = myDataset;
sContext = context;
}
@Override
public MyAdapter.ViewHolder onCreateViewHolder(ViewGroup parent,int viewType) {
View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.my_text_view, parent, false);
ViewHolder holder = new ViewHolder(v);
holder.mNameTextView.setOnClickListener(MyAdapter.this);
holder.mNameTextView.setOnLongClickListener(MyAdapter.this);
holder.mNameTextView.setTag(holder);
return holder;
}
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
holder.mNameTextView.setText(mDataset.get(position));
}
@Override
public int getItemCount() {
return mDataset.size();
}
@Override
public void onClick(View view) {
ViewHolder holder = (ViewHolder) view.getTag();
if (view.getId() == holder.mNameTextView.getId()) {
Toast.makeText(sContext, holder.mNameTextView.getText(), Toast.LENGTH_SHORT).show();
}
}
@Override
public boolean onLongClick(View view) {
ViewHolder holder = (ViewHolder) view.getTag();
if (view.getId() == holder.mNameTextView.getId()) {
mDataset.remove(holder.getPosition());
notifyDataSetChanged();
Toast.makeText(sContext, "Item " + holder.mNameTextView.getText() + " has been removed from list",
Toast.LENGTH_SHORT).show();
}
return false;
}
public static class ViewHolder extends RecyclerView.ViewHolder {
public TextView mNumberRowTextView;
public TextView mNameTextView;
public ViewHolder(View v) {
super(v);
mNameTextView = (TextView) v.findViewById(R.id.nameTextView);
}
}
}
мой макет:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:gravity="center_vertical"
android:id="@+id/layout">
<TextView
android:id="@+id/nameTextView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="18sp"
android:padding="5dp"
android:background="@drawable/greyline"/>
<ImageView
android:id="@+id/crossButton"
android:layout_width="16dp"
android:layout_height="16dp"
android:visibility="gone"
android:layout_marginLeft="50dp"
android:src="@drawable/cross" />
</LinearLayout>
Как я могу получить что-то вроде onClick, работающего для моего crossButton ImageView? Есть лучший способ? Может быть, изменение всего элемента onclick в удалить элемент? В окне recyclerview отображается список местоположений, которые необходимо изменить. Любые технические рекомендации или замечания / предложения по наилучшей реализации будут высоко оценены.
12 ответов:
Я сделал что-то подобное. В вашем
MyAdapter
:public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener{ public CardView mCardView; public TextView mTextViewTitle; public TextView mTextViewContent; public ImageView mImageViewContentPic; //...... public ImageView imgViewRemoveIcon; public ViewHolder(View v) { super(v); mCardView = (CardView) v.findViewById(R.id.card_view); mTextViewTitle = (TextView) v.findViewById(R.id.item_title); mTextViewContent = (TextView) v.findViewById(R.id.item_content); mImageViewContentPic = (ImageView) v.findViewById(R.id.item_content_pic); //...... imgViewRemoveIcon = (ImageView) v.findViewById(R.id.remove_icon); mTextViewContent.setOnClickListener(this); imgViewRemoveIcon.setOnClickListener(this); v.setOnClickListener(this); mTextViewContent.setOnLongClickListener(new View.OnLongClickListener() { @Override public boolean onLongClick(View view) { if (mItemClickListener != null) { mItemClickListener.onItemClick(view, getPosition()); } return false; } }); } @Override public void onClick(View v) { //Log.d("View: ", v.toString()); //Toast.makeText(v.getContext(), mTextViewTitle.getText() + " position = " + getPosition(), Toast.LENGTH_SHORT).show(); if(v.equals(imgViewRemoveIcon)){ removeAt(getPosition()); }else if (mItemClickListener != null) { mItemClickListener.onItemClick(v, getPosition()); } } } public void setOnItemClickListener(final OnItemClickListener mItemClickListener) { this.mItemClickListener = mItemClickListener; } public void removeAt(int position) { mDataset.remove(position); notifyItemRemoved(position); notifyItemRangeChanged(position, mDataSet.size()); }
надеюсь, что это помогает.
Edit:
getPosition()
является устаревшим теперь использовать .
во-первых, элемент должен быть удален из списка!
mDataSet.remove(getAdapterPosition());
затем:
notifyItemRemoved(getAdapterPosition()); notifyItemRangeChanged(getAdapterPosition(),mDataSet.size());
если все еще элемент не удален используйте этот магический метод:)
private void deleteItem(int position) { mDataSet.remove(position); notifyItemRemoved(position); notifyItemRangeChanged(position, mDataSet.size()); holder.itemView.setVisibility(View.GONE); }
Я пробовал все вышеперечисленные ответы, но вставка или удаление элементов в recyclerview вызывает проблему с позицией в наборе данных. В конечном итоге с помощью
delete(getAdapterPosition());
внутри viewHolder, который отлично работал при поиске положения предметов.
на самом деле с ранее упомянутым кодом (который вы можете найти здесь ниже), все RecyclerView childs анимируются, когда удаляется один элемент, и это не отзывчивое решение вообще, также потому, что у него есть некоторые потенциальные скрытые проблемы
public void remove(int position) { dataset.remove(position); notifyItemRemoved(position); notifyItemRangeChanged(position, mDataSet.size()); }
вместо этого следующий фрагмент будет воспроизводить анимацию только на ребенке, который удаляется, и с моей стороны он перестал бросать
IndexOutOfBoundException
s, помеченные отладчиком как "несоответствие данных"void remove(int position) { dataset.removeAt(position); notifyItemChanged(position); notifyItemRangeRemoved(position, 1); }
если мы копать в
RecyclerView
класс мы можем найти pice javadoc, который имеет объяснение, действительно, в качестве второго параметра мы должны передать количество элементов, которые удаляются из набора данных, а не общее количество элементов/** * Notify any registered observers that the <code>itemCount</code> items previously * located at <code>positionStart</code> have been removed from the data set. The items * previously located at and after <code>positionStart + itemCount</code> may now be found * at <code>oldPosition - itemCount</code>. * * <p>This is a structural change event. Representations of other existing items in the data * set are still considered up to date and will not be rebound, though their positions * may be altered.</p> * * @param positionStart Previous position of the first item that was removed * @param itemCount Number of items removed from the data set */ public final void notifyItemRangeRemoved(int positionStart, int itemCount) { mObservable.notifyItemRangeRemoved(positionStart, itemCount); }
проблема, с которой я столкнулся, заключалась в том, что я удалял элемент из списка, который больше не был связан с адаптером, чтобы убедиться, что вы изменяете правильный адаптер, который вы можете реализовать такой метод в своем адаптере:
public void removeItemAtPosition(int position) { items.remove(position); }
и назовите его в своем фрагменте или активности следующим образом:
adapter.removeItemAtPosition(position);
возможно, дубликат ответа, но довольно полезно для меня. Вы можете реализовать метод, приведенный ниже в
RecyclerView.Adapter<RecyclerView.ViewHolder>
и может использовать этот метод в соответствии с вашими требованиями, я надеюсь, что это будет работать для васpublic void removeItem(@NonNull Object object) { mDataSetList.remove(object); notifyDataSetChanged(); }
вот некоторые визуальные дополнительные примеры. Смотрите мой более полный ответ для примеров добавления и удаления диапазона.
добавить один элемент
добавить "свинья" на индекс
2
.String item = "Pig"; int insertIndex = 2; data.add(insertIndex, item); adapter.notifyItemInserted(insertIndex);
удалить один элемент
удалить "свинью" из списка.
int removeIndex = 2; data.remove(removeIndex); adapter.notifyItemRemoved(removeIndex);
если вы хотите удалить пункт, вы должны сделать это: сначала удалить элемент:
phones.remove(position);
на следующем шаге вы должны уведомить ваш адаптер recycler, что вы удалите элемент с помощью этого кода:
notifyItemRemoved(position); notifyItemRangeChanged(position, phones.size());
но если вы измените пункт, сделать это: сначала измените параметр вашего объекта следующим образом:
Service s = services.get(position); s.done = "Cancel service"; services.set(position,s);
или так :
Service s = new Service(); services.set(position,s);
затем сообщите своему адаптеру recycler, что вы изменяете элемент с помощью этого кода:
notifyItemChanged(position); notifyItemRangeChanged(position, services.size());
надеюсь, что помогает вы.
//////// set the position holder.cancel.setTag(position); ///// click to remove an item from recycler view and an array list holder.cancel.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { int positionToRemove = (int)view.getTag(); //get the position of the view to delete stored in the tag mDataset.remove(positionToRemove); notifyDataSetChanged(); } });
сделать интерфейс в пользовательский класс адаптера и обработки нажмите событие на recycler view..
onItemClickListner onItemClickListner; public void setOnItemClickListner(CommentsAdapter.onItemClickListner onItemClickListner) { this.onItemClickListner = onItemClickListner; } public interface onItemClickListner { void onClick(Contact contact);//pass your object types. } @Override public void onBindViewHolder(ItemViewHolder holder, int position) { // below code handle click event on recycler view item. holder.itemView.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { onItemClickListner.onClick(mContectList.get(position)); } }); }
после определения адаптера и привязки к виду recycler вызывается ниже кода..
adapter.setOnItemClickListner(new CommentsAdapter.onItemClickListner() { @Override public void onClick(Contact contact) { contectList.remove(contectList.get(contectList.indexOf(contact))); adapter.notifyDataSetChanged(); } }); }