programing

Android Recycleer View 아이템 추가 및 삭제

minecode 2022. 12. 10. 10:52
반응형

Android Recycleer View 아이템 추가 및 삭제

RecycleerView에는 TextView텍스트 박스와 십자버튼 ImageView가 있습니다.recyclerview 밖에 있는 버튼이 있어 십자버튼 ImageView가 표시되거나 사라집니다.

아이템의 크로스버튼 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>

crossButton ImageView에서 onClick과 같은 기능을 사용하려면 어떻게 해야 합니까?더 좋은 방법이 있을까요?클릭해서 아이템 전체를 삭제로 변경하는 것은 어떨까요?재사용 보기는 편집해야 하는 위치 목록을 표시합니다.최선의 구현에 관한 기술적인 조언이나 코멘트/제안해주시면 감사하겠습니다.

도도비비비비비을을이있있요요요요요요 。고객님의 고객명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()★★★★★★ 。

우선, 아이템은 리스트에서 삭제되어야 합니다!

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

코틀린 버전

private fun deleteItem(position: Int) {
    mDataSet.removeAt(position)
    notifyItemRemoved(position)
    notifyItemRangeChanged(position, mDataSet.size)
    holder.itemView.visibility = View.GONE
}

문제

RecyclerView는 데이터를 효율적이고 응답성 있게 표시하도록 구축되었습니다.일반적으로 어댑터로 전달된 데이터 집합이 있고 데이터를 표시하기 위해 루프 스루됩니다.츠키다

private ArrayList<String> mDataset;

요점은 RecycleerView가 데이터 세트에 연결되어 있지 않기 때문에 데이터 세트의 변경을 인식하지 못한다는 것입니다.

데이터를 한 번만 읽고 View에 표시보유자이지만 데이터 집합의 변경 내용이 UI로 전파되지 않습니다.

즉, 데이터 목록에서 삭제/추가 작업을 수행할 때마다 변경 내용이 RecyclerView에 직접 반영되지 않습니다(인덱스 5에서 항목을 삭제해도 6번째 요소는 Recycler View에 그대로 남습니다).

(구식) 솔루션

RecyclerView에서는 데이터 집합 변경 내용을 전달하기 위한 몇 가지 방법을 보여 줍니다.이러한 변경 내용은 목록 항목에 직접 반영됩니다.

표준 Android API를 사용하면 데이터 제거 프로세스를 View 제거 프로세스와 바인딩할 수 있습니다.

여기서 설명하는 방법은 다음과 같습니다.

notifyItemChanged(index: Int)
notifyItemInserted(index: Int)
notifyItemRemoved(index: Int)
notifyItemRangeChanged(startPosition: Int, itemCount: Int)
notifyItemRangeInserted(startPosition: Int, itemCount: Int)
notifyItemRangeRemoved(startPosition: Int, itemCount: Int)

완전한 (구식)솔루션

항목을 추가, 변경 또는 삭제할 때마다 발생하는 작업을 적절하게 지정하지 않으면RecyclerView목록 항목의 애니메이션은 목록 주위에 있는 여러 보기를 이동하는 방법에 대한 정보가 부족하기 때문에 응답하지 않습니다.

할 수 사이드로 RecycleerView는 것을 합니다).IndexOutOfBoundException스택 트레이스에 의해 「데이터 불일치」라고 마크됩니다).

void remove(position: Int) {
    dataset.removeAt(position)
    notifyItemChanged(position)
    notifyItemRangeRemoved(position, 1)
}

밑에 요.RecyclerView을 설명하는 .notifyItemRangeRemoved 는, 데이터 세트로부터 삭제되는 항목의 수가 아니고, 항목의 합계수가 아닙니다(다른 정보원에서는 잘못 보고되고 있습니다).

    /**
     * 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);
    }

오픈 소스 솔루션

FastAdapter, Epoxy 또는 Groupie와 같은 라이브러리가 비즈니스를 처리하도록 할 수 있습니다.또, 데이터 바인딩이 있는 관측 가능한 리사이클러 뷰를 사용할 수도 있습니다.

새 목록 어댑터

구글은 최근 리사이클러 뷰 어댑터를 쓰는 새로운 방법을 선보였는데, 이 어댑터는 매우 잘 작동하고 반응성 있는 데이터를 지원합니다.

이것은 새로운 접근 방식이고 약간의 리팩터링이 필요하지만 모든 것이 더 부드러워지기 때문에 100% 바꿀 가치가 있습니다.

여기 서류가 있고, 여기에 그것을 설명하는 중간 기사가 있습니다.

다음은 시각적인 보충 예시입니다.범위 추가 및 삭제의 예에 대해서는 자세한 답변을 참조하십시오.

단일 항목 추가

"를 추가합니다2.

단일 항목 삽입

String item = "Pig";
int insertIndex = 2;
data.add(insertIndex, item);
adapter.notifyItemInserted(insertIndex);

단일 항목 제거

목록에서 "돼지"를 제거합니다.

단일 항목 제거

int removeIndex = 2;
data.remove(removeIndex);
adapter.notifyItemRemoved(removeIndex);

아마도 중복된 답변이지만 나에게 꽤 유용하다.은 다음에 방법으로 실행할 수 .RecyclerView.Adapter<RecyclerView.ViewHolder> 이 할 수 . 저는 그것이 당신에게 효과가 있기를 바랍니다.

public void removeItem(@NonNull Object object) {
        mDataSetList.remove(object);
        notifyDataSetChanged();
    }

위의 답변을 모두 시도해 보았습니다만, recycerview에 아이템을 삽입 또는 삭제하면 dataset의 위치에 문제가 발생합니다.결국 사용되었습니다.delete(getAdapterPosition());★★★★★에 큰 이 된 .아이템의 위치를 찾는 데 큰 도움이 되는 홀더.

문제는 어댑터에 다음과 같은 방법을 구현할 수 있는 올바른 어댑터를 수정하고 있는지 확인하기 위해 어댑터와 더 이상 관련이 없는 항목을 목록에서 삭제하는 것이었습니다.

public void removeItemAtPosition(int position) {
    items.remove(position);
}

fragment 또는 액티비티에서 다음과 같이 호출합니다.

adapter.removeItemAtPosition(position);
  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 txtValue;
    TextView txtCategory;
    ImageView imgInorEx;
    ImageView imgCategory;
    TextView txtDate;

    public MyViewHolder(@NonNull View itemView) {
        super(itemView);
        txtValue= itemView.findViewById(R.id.id_values);
        txtCategory= itemView.findViewById(R.id.id_category);
        imgInorEx= itemView.findViewById(R.id.id_inorex);
        imgCategory= itemView.findViewById(R.id.id_imgcategory);
        txtDate= itemView.findViewById(R.id.id_date);
    }
}

@NonNull
@Override
public void onBindViewHolder(@NonNull final MyViewHolder myViewHolder, int i) {

    myViewHolder.txtValue.setText(String.valueOf(list.get(i).getValuee()));
    myViewHolder.txtCategory.setText(list.get(i).getCategory());
    myViewHolder.imgInorEx.setBackgroundColor(list.get(i).getImg_inorex());
    myViewHolder.imgCategory.setImageResource(list.get(i).getImg_category());
    myViewHolder.txtDate.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();
}}      

이게 도움이 됐으면 좋겠네요

항목을 제거하려면 다음을 수행해야 합니다. 먼저 항목을 제거하십시오.

phones.remove(position);

다음 단계에서는 다음 코드로 아이템을 삭제했음을 재활용 어댑터에게 통지해야 합니다.

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

희망이 당신을 돕습니다.

이 항목을 arrayList의 이 항목을 제거해야 합니다.

myDataset.remove(holder.getAdapterPosition());
notifyItemRemoved(holder.getAdapterPosition());
notifyItemRangeChanged(holder.getAdapterPosition(), getItemCount());
  String str = arrayList.get(position);
  arrayList.remove(str);
  MyAdapter.this.notifyDataSetChanged();

To 메서드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();
            }
        });

Adapter 클래스가 아닌 Main 클래스에서 이와 같은 기능을 구현하고 싶은 경우 다음을 사용할 수 있습니다.

public void removeAt(int position) {
    peopleListUser.remove(position);

    friendsListRecycler.getAdapter().notifyItemRemoved(position);
    friendsListRecycler.getAdapter().notifyItemRangeChanged(position, peopleListUser.size());
}

friends List Recycler는 어댑터 이름입니다.

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

인터페이스를 커스텀 어댑터 클래스로 만들고 재활용기 뷰에서 클릭 이벤트를 처리합니다.

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

예를 들어 method getadapter position()의 어댑터 위치는 뷰홀더 오브젝트에 있습니다.그러니까 코드를 이렇게 넣으셔야 돼요.

mdataset.remove(holder.getadapterposition());

액티비티:

mAdapter.updateAt(pos, text, completed);
mAdapter.removeAt(pos);

어댑터 내:

void removeAt(int position) {
    list.remove(position);
    notifyItemRemoved(position);
    notifyItemRangeChanged(position, list.size());
}

void updateAt(int position, String text, Boolean completed) {
    TodoEntity todoEntity = list.get(position);
    todoEntity.setText(text);
    todoEntity.setCompleted(completed);
    notifyItemChanged(position);
}

언급URL : https://stackoverflow.com/questions/26076965/android-recyclerview-addition-removal-of-items

반응형