Android RecyclerView добавление и удаление элементов
У меня есть RecyclerView с текстовым полем TextView и крест-кнопкой ImageView. У меня есть кнопка вне recyclerview, которая делает крест-кнопку ImageView видимой/удаленной.
Я ищу, чтобы удалить элемент из recylerview, когда нажата кнопка CrossView 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 показан список мест, которые необходимо отредактировать. Любые технические консультации или комментарии/предложения по наилучшей реализации будут чрезвычайно оценены.
Ответы
Ответ 1
Я сделал что-то подобное. В вашем 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());
}
Надеюсь это поможет.
Редактировать:
getPosition()
устарела, используйте getAdapterPosition()
.
Ответ 2
прежде всего, элемент должен быть удален из списка!
mDataSet.remove(getAdapterPosition());
то
notifyItemRemoved(getAdapterPosition());
notifyItemRangeChanged(getAdapterPosition(),mDataSet.size());
Ответ 3
если элемент, который не был удален, использует этот магический метод:)
private void deleteItem(int position) {
mDataSet.remove(position);
notifyItemRemoved(position);
notifyItemRangeChanged(position, mDataSet.size());
holder.itemView.setVisibility(View.GONE);
}
Ответ 4
Возможно, дублирующий ответ, но весьма полезный для меня. Вы можете реализовать метод, приведенный ниже, в RecyclerView.Adapter<RecyclerView.ViewHolder>
и можете использовать этот метод в соответствии с вашими требованиями, я надеюсь, что он сработает для вас.
public void removeItem(@NonNull Object object) {
mDataSetList.remove(object);
notifyDataSetChanged();
}
Ответ 5
На самом деле с ранее упомянутым кодом (который вы можете найти здесь ниже) все дочерние элементы RecyclerView анимируются, когда удаляется один элемент, и это вообще не адаптивное решение, в том числе из-за потенциальных скрытых проблем.
public void remove(int position) {
dataset.remove(position);
notifyItemRemoved(position);
notifyItemRangeChanged(position, mDataSet.size());
}
Вместо этого следующий фрагмент будет воспроизводить анимацию только для удаляемого дочернего элемента, и с моей стороны он перестал генерировать IndexOutOfBoundException
, помеченные отладчиком как "несоответствие данных".
void remove(int position) {
dataset.removeAt(position);
notifyItemChanged(position);
notifyItemRangeRemoved(position, 1);
}
Если мы покопаемся в классе RecyclerView
то сможем найти кусок 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);
}
Ответ 6
Я попробовал все вышеперечисленные ответы, но вставка или удаление элементов для recyclerview вызывает проблему с положением в наборе данных. Закончилось использование delete(getAdapterPosition());
внутри viewHolder, который отлично справился с поиском позиции элементов.
Ответ 7
Проблема была в том, что я удалял элемент из списка, который больше не ассоциировался с адаптером, чтобы убедиться, что вы изменяете правильный адаптер, который вы можете реализовать в адаптере таким образом:
public void removeItemAtPosition(int position) {
items.remove(position);
}
И назовите его в своем фрагменте или действии следующим образом:
adapter.removeItemAtPosition(position);
Ответ 8
Вот несколько наглядных дополнительных примеров. Смотрите мой более полный ответ для примеров добавления и удаления диапазона.
Добавить один элемент
Добавить "Pig" в указателе 2
.
![Insert single item]()
String item = "Pig";
int insertIndex = 2;
data.add(insertIndex, item);
adapter.notifyItemInserted(insertIndex);
Удалить один элемент
Удалить "Свинью" из списка.
![Remove single item]()
int removeIndex = 2;
data.remove(removeIndex);
adapter.notifyItemRemoved(removeIndex);
Ответ 9
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> {
private Context context;
private List<cardview_widgets> list;
public MyAdapter(Context context, List<cardview_widgets> list) {
this.context = context;
this.list = list;
}
@NonNull
@Override
public MyViewHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int i) {
View view = LayoutInflater.from(this.context).inflate(R.layout.fragment1_one_item,
viewGroup, false);
return new MyViewHolder(view);
}
public static class MyViewHolder extends RecyclerView.ViewHolder {
TextView txt_value;
TextView txt_category;
ImageView img_inorex;
ImageView img_category;
TextView txt_date;
public MyViewHolder(@NonNull View itemView) {
super(itemView);
txt_value = itemView.findViewById(R.id.id_values);
txt_category = itemView.findViewById(R.id.id_category);
img_inorex = itemView.findViewById(R.id.id_inorex);
img_category = itemView.findViewById(R.id.id_imgcategory);
txt_date = itemView.findViewById(R.id.id_date);
}
}
@NonNull
@Override
public void onBindViewHolder(@NonNull final MyViewHolder myViewHolder, int i) {
myViewHolder.txt_value.setText(String.valueOf(list.get(i).getValuee()));
myViewHolder.txt_category.setText(list.get(i).getCategory());
myViewHolder.img_inorex.setBackgroundColor(list.get(i).getImg_inorex());
myViewHolder.img_category.setImageResource(list.get(i).getImg_category());
myViewHolder.txt_date.setText(list.get(i).getDate());
myViewHolder.itemView.setOnLongClickListener(new View.OnLongClickListener() {
@Override
public boolean onLongClick(View v) {
list.remove(myViewHolder.getAdapterPosition());
notifyDataSetChanged();
return false;
}
});
}
@Override
public int getItemCount() {
return list.size();
}}
Я надеюсь, что это поможет вам.
Ответ 10
String str = arrayList.get(position);
arrayList.remove(str);
MyAdapter.this.notifyDataSetChanged();
Ответ 11
если вы хотите удалить элемент, вы должны сделать это:
сначала удалите элемент:
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);
затем сообщите об этом адаптеру вашего ресайклера, который вы модифицируете по этому коду:
notifyItemChanged(position);
notifyItemRangeChanged(position, services.size());
надеюсь, вам поможет.
Ответ 12
Incase Любой, кто хочет реализовать нечто подобное в классе Main вместо класса Adapter, вы можете использовать 'public void removeAt (int position) {peopleListUser.remove(position);
friendsListRecycler.getAdapter().notifyItemRemoved(position);
friendsListRecycler.getAdapter().notifyItemRangeChanged(position, peopleListUser.size());
}'
где friendsListRecycler - это имя адаптера
Ответ 13
//////// 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();
}
});
Ответ 14
сделать интерфейс в пользовательский класс адаптера и обработать событие click в представлении переработчика.
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));
}
});
}
после определения адаптера и привязки к представлению переработчика, называемому ниже кода.
adapter.setOnItemClickListner(new CommentsAdapter.onItemClickListner() {
@Override
public void onClick(Contact contact) {
contectList.remove(contectList.get(contectList.indexOf(contact)));
adapter.notifyDataSetChanged();
}
});
}
Ответ 15
Метод onBindViewHolder
написать этот код
holder.remove.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Cursor del=dbAdapter.ExecuteQ("delete from TblItem where Id="+values.get(position).getId());
values.remove(position);
notifyDataSetChanged();
}
});
Ответ 16
сослаться https://www.youtube.com/watch?v=HMjI7cLsyfw на этот... он работал для меня