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 98

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());
}

вместо этого следующий фрагмент будет воспроизводить анимацию только на ребенке, который удаляется, и с моей стороны он перестал бросать IndexOutOfBoundExceptions, помеченные отладчиком как "несоответствие данных"

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.

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);
  String str = arrayList.get(position);
  arrayList.remove(str);
  MyAdapter.this.notifyDataSetChanged();

если вы хотите удалить пункт, вы должны сделать это: сначала удалить элемент:

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();
        }
    });
}