删除 RecyclerView 项目

新手上路,请多包涵

我有一个 RecyclerView,它在每个特定时间段添加项目。添加对象时,如果列表项超过(比方说)500,则第一个项目将被删除并添加新项目。

如果 RecyclerView 不能再向下滚动 !recyclerView.canScrollVertically(1); 然后,在添加新项目后,RecyclerView 将 smoothScroolToPosition() 到最后一个位置。

问题出在哪里? 好的,如果 ReyclerView 在移除旧项目时位于中间(不是底部,不是顶部),它会突然向上跳一些位置。我希望 RecyclerView 在移除顶部的项目时不要跳转位置并保持原位

我尝试使用 layoutManager.setStackFromEnd(true); 但没有运气

有什么建议么?

部分代码(无关代码已删除)。当 RecyclerView 从中间显示列表项并按 VOLUME_UP 时,可以重现我的问题:

 public class ActivityMain extends ActionBarActivity {

    public static final int MAX_LOG_ITEMS = 500;

    private RecyclerView mRecyclerView;
    private AdapterLog mRecyclerAdapter;
    private boolean mAutoScroll = true;

    private DataReceiver mDataReceiver;
    private Handler mLogHandler = new Handler() {
        @Override public void handleMessage(Message msg) {
            switch (msg.what){
                case DataReceiver.CAT_LOGS:
                    List<Log> catLogs = (List<Log>) msg.obj;
                    updateLogs(catLogs);
                    break;
                case DataReceiver .CLEAR_LOGS:
                    if (mRecyclerAdapter.getItemCount() > MAX_LOG_ITEMS)
                        mRecyclerAdapter.removeFirstItems(mRecyclerAdapter.getItemCount() - MAX_LOG_ITEMS);
                    break;
                case Logcat.REMOVE_LOGS:
                    mRecyclerAdapter.clear();
            }
        }

    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);

        final LinearLayoutManager layoutManager = new LinearLayoutManager(this);
        layoutManager.setStackFromEnd(true);
        mRecyclerAdapter = new AdapterLog();

        mRecyclerView = (RecyclerView) findViewById(R.id.activity_main_recyclerview);
        mRecyclerView.setHasFixedSize(true);
        mRecyclerView.setLayoutManager(layoutManager);
        mRecyclerView.setAdapter(mRecyclerAdapter);
        mRecyclerView.setOnScrollListener(new UIUtils.ScrollManager(toolbarContainer != null ?
                toolbarContainer : toolbar){
                @Override public void onScrolled(RecyclerView r, int dx, int dy) {
                    super.onScrolled(r, dx, dy);
                    mAutoScroll = !r.canScrollVertically(1);
                    }
                });
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        switch(keyCode){
            case KeyEvent.KEYCODE_VOLUME_UP:
//                mAutoScroll = false;
//                mRecyclerView.scrollToPosition(0);
//                if (mRecyclerAdapter.getItemCount() > MAX_LOG_ITEMS)
                    mRecyclerAdapter.removeFirstItems(mRecyclerAdapter.getItemCount() - 50);

                return true;
            case KeyEvent.KEYCODE_VOLUME_DOWN:
                mAutoScroll = true;
                mRecyclerView.scrollToPosition(mRecyclerAdapter.getItemCount() -1);
                return true;
        }
        return false;
    }

    private void updateLogs(final List<Log> logList) {
        final boolean scroll = mAutoScroll;
        mRecyclerAdapter.addAll(logList);
        if (scroll) mRecyclerView.smoothScrollToPosition(mRecyclerAdapter.getItemCount() - 1);
    }

}

回收适配器:

 public class AdapterLog extends RecyclerView.Adapter<AdapterLog.ViewHolder> {

    private final List<Log> mLogList;

    public AdapterLog() {
        this.mLogList = new ArrayList<Log>();
    }

    @Override
    public AdapterLog.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View v = LayoutInflater.from(parent.getContext())
            .inflate(R.layout.listitem_log, parent, false);
        return new ViewHolder(v);
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, int position) {
        holder.mTextView.setText(getItem(position).getMessage());
        holder.mTextView.setTextColor(getItem(position).getLevel().getColor());
    }

    @Override
    public int getItemCount() {
        return mLogList.size();
    }

    public Log getItem(int position) {
        return mLogList.get(position);
    }

    public void addAll(List<Log> logList) {
        mLogList.addAll(logList);
        notifyDataSetChanged();
    }

    public void removeFirstItems(int count) {
        for (int i=0; i<count; i++) mLogList.remove(0);
        notifyDataSetChanged();
    }

    public void clear() {
        mLogList.clear();
        notifyDataSetChanged();
    }

    public static class ViewHolder extends RecyclerView.ViewHolder {
        public TextView mTextView;
        public ViewHolder(View v) {
            super(v);
            mTextView = (TextView) v.findViewById(R.id.listitem_log_textview);
        }
    }
}

原文由 BamsBamx 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 391
2 个回答

你告诉你的观点你改变了你的整个适配器。对于添加、删除和重新排序,请考虑使用以下方法:

 notifyItemRangeChanged
notifyItemRangeInserted
notifyItemRangeRemoved
notifyItemMoved
notifyItemInserted
notifyItemChanged
notifyItemRemoved

因此,在您的情况下,您要删除顶部 count 项目,因此:

 notifyItemRangeRemoved(0,count)

应该做的工作。

原文由 Pedro Oliveira 发布,翻译遵循 CC BY-SA 3.0 许可协议

 public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {

    private McPurchasePricingCellBinding itemBinding;
    private OnProductItemListener listener;

    public ViewHolder(McPurchasePricingCellBinding itemBinding, OnProductItemListener listener) {
        super(itemBinding.getRoot());
        this.itemBinding = itemBinding;
        this.listener = listener;
        itemBinding.removeCartContainer.setOnClickListener(this);

    }

    @Override
    public void onClick(View v) {
        if (v.getId() == R.id.remove_cart_container) {
            if (listener == null) {
                return;
            }
            int position = getAdapterPosition();
            if (position != RecyclerView.NO_POSITION) {
                listener.onRemoveItem(position);
            }
        }
    }
}

public void removeAt(int position) {
    arrayList.remove(position);
    notifyItemRemoved(position);
}
 public interface OnProductItemListener {
    void onRemoveItem(int position);
}

原文由 Azharul Islam 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题