Android: how to add items and refresh list when using PagedList?
Asked Answered
U

1

6

I've used PagedList for loading page of items and display them in RecyclerView, and it is successfully displaying list of numbers(0,1,2,3....,99), the list can always be scroll up and down without any problem. When I tried to click a button to call appendItems() to append new data, its still fine. However, the app crashed when I scroll the list.

java.util.ConcurrentModificationException
    at java.util.AbstractList$SubAbstractList.size(AbstractList.java:360)
    at androidx.paging.PagedStorage.get(PagedStorage.java:153)
    at androidx.paging.PagedList.get(PagedList.java:384)
    at androidx.paging.AsyncPagedListDiffer.getItem(AsyncPagedListDiffer.java:206)
    at androidx.paging.PagedListAdapter.getItem(PagedListAdapter.java:156)
    at com.mysample.pagedlist.MyPagedListAdapter.onBindViewHolder(MyPagedListAdapter.java:38)
    at com.mysample.pagedlist.MyPagedListAdapter.onBindViewHolder(MyPagedListAdapter.java:19)

It seems I don't know the right way to append more data and refresh the display. Can anyone tell me how to work it out?

Api is served as database and dao

public class Api {
    private static List<String> list = null;

    private static void initListIfNeeded() {
        if (list == null) {
            list = new ArrayList<>();
            for (int i = 0; i < 100; i++) {
                list.add("" + i);
            }
        }
    }

    public static List<String> pageItems(int page, int pageSize) {
        initListIfNeeded();
        int start = page * pageSize;
        int end = (page + 1) * pageSize;
        if (end > list.size()) {
            end = list.size();
        }
        if (page < 0 || start >= list.size()) {
            return new ArrayList<>();
        } else {
            return list.subList(start, end);
        }
    }

    public static void appendItems() {
        initListIfNeeded();
        int size = list.size();
        for (int i = 0; i < 10; i++) {
            list.add("" + (size + i));
        }
    }
}

DataSource used for handling pagination of data

public class MyDataSource extends PageKeyedDataSource<Integer, String> {

    public MyDataSource() {
    }

    @Override
    public void loadInitial(@NonNull LoadInitialParams<Integer> params, @NonNull LoadInitialCallback<Integer, String> callback) {
        callback.onResult(Api.pageItems(0, params.requestedLoadSize), -1, 1);
    }

    @Override
    public void loadBefore(@NonNull LoadParams<Integer> params, @NonNull LoadCallback<Integer, String> callback) {
        callback.onResult(Api.pageItems(params.key, params.requestedLoadSize), params.key - 1);
    }

    @Override
    public void loadAfter(@NonNull LoadParams<Integer> params, @NonNull LoadCallback<Integer, String> callback) {
        callback.onResult(Api.pageItems(params.key, params.requestedLoadSize), params.key + 1);
    }

}

The corresponding ViewModel class

public class MyViewModel extends ViewModel {

    public LiveData<PagedList<String>> list;

    public MyViewModel() {
        list = new LivePagedListBuilder<>(new DataSource.Factory<Integer, String>() {
            @Override
            public DataSource<Integer, String> create() {
                return new MyDataSource();
            }
        },  new PagedList.Config.Builder().setPageSize(20).setInitialLoadSizeHint(20).build()).build();
    }

    public void appendItems() {
        Api.appendItems();
    }
}

The activity

public class PagedListActivity extends AppCompatActivity {

    private MyPagedListAdapter adapter;
    private MyViewModel viewModel;
    private RecyclerView recyclerView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_paged_list);
        adapter = new MyPagedListAdapter(LayoutInflater.from(this));

        recyclerView = findViewById(R.id.recyclerView);
        recyclerView.setLayoutManager(new LinearLayoutManager(this));
        recyclerView.setAdapter(adapter);
        viewModel = ViewModelProviders.of(this).get(MyViewModel.class);
        viewModel.list.observe(this, new Observer<PagedList<String>>() {
            @DebugLog
            @Override
            public void onChanged(PagedList<String> strings) {
                adapter.submitList(strings);
            }
        });
    }

    @DebugLog
    public void onClick(View view) {
        viewModel.appendItems();
    }
}

The adapter

public class MyPagedListAdapter extends PagedListAdapter<String, MyPagedListAdapter.VH> {

    private LayoutInflater inflater;

    protected MyPagedListAdapter(LayoutInflater inflater) {
        super(DIFF_CALLBACK);
        this.inflater = inflater;
    }

    @NonNull
    @Override
    public VH onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        View view = inflater.inflate(R.layout.item_text, parent, false);
        return new VH(view);
    }

    @Override
    public void onBindViewHolder(@NonNull VH holder, int position) {
        holder.setData(getItem(position));
    }

    public class VH extends RecyclerView.ViewHolder {

        private final TextView txt;

        public VH(@NonNull View itemView) {
            super(itemView);
            txt = itemView.findViewById(R.id.txt);
        }

        public void setData(String text) {
            txt.setText(text);
        }
    }

    private static DiffUtil.ItemCallback<String> DIFF_CALLBACK = new DiffUtil.ItemCallback<String>() {
        @Override
        public boolean areItemsTheSame(@NonNull String oldItem, @NonNull String newItem) {
            return oldItem.equalsIgnoreCase(newItem);
        }

        @Override
        public boolean areContentsTheSame(@NonNull String oldItem, @NonNull String newItem) {
            return oldItem.equalsIgnoreCase(newItem);
        }
    };

}
Uriiah answered 19/10, 2018 at 11:43 Comment(0)
R
2

From this stacktrace:

java.util.ConcurrentModificationException
    at java.util.AbstractList$SubAbstractList.size(AbstractList.java:360)

we can tell that

a) app crashes when someone is trying to modify list while iteration over list is in process
b) this happens on SubAbstractList instance (that is created when you call list.subList(start, end);

App crashes because sublist created before original list modifications is no longer valid, after you modify (append items to) originial list

Instead of returning list.subList(start, end); from Api#pageItems try
returning new ArrayList<>(list.subList(start, end));

For RecyclerView to display your changes, you should call Adapter#notifyItemRangeInserted or RW adapter

Resign answered 19/10, 2018 at 11:54 Comment(3)
thanks, the modification did fix the error, however, the added items are not displaying in the list, what should I do to refresh the list?Uriiah
by adding adapter.notifyItemRangeInserted(some_start_pos, some_count) after viewModel.appendItems() in activity , it allows list showing more items when the list isn't scrolled to end. If the list is scrolled to the end, and I click button to append items, nothing happens; no more items are shownUriiah
very late to this by @Uriiah did you ever find a solution to this?Ebbie

© 2022 - 2024 — McMap. All rights reserved.