update RecyclerView with Android LiveData

36,547

Solution 1

Like @Lyla said, you should observe the whole list as LiveData in Fragment or Activity, when receive changes, you should set the whole list to the adapter by DiffUtil.

Fake code:

PostViewModel {
    LiveData<List<Post>> posts;  // posts comes from DAO or Webservice
}

MyFragment extends LifecycleFragment {
    PostAdapter postAdapter;

    ...

    void onActivityCreated() {
        ...
        postViewModel.posts.observer(this, (postList) -> {
            postAdapter.setPosts(postList);
        }
    }       
}

PostAdapter {
    void setPosts(List<Post> postList) {
        DiffUtil.DiffResult result = DiffUtil.calculateDiff(new DiffUtil.Callback() {...}
        ...
    }
}

Solution 2

Using DiffUtil might help with updating one row in a huge list. You can then have LiveData wrap the list of comments instead of a single comment or attribute of a comment.

Here's an example of using DiffUtil within a RecyclerView adapter and the list LiveData observation code in the fragment.

Solution 3

Use Transformations.switchMap() to swap the underlying Post object. Then there is no need to remove and re-add observers when the cell is recycled.

@Override
public void onBindViewHolder(PostViewHolder vh, int position) {
    Post post = getPost(position);
    vh.bind(post);
}

Then in your ViewHolder class

public class PostViewHolder extends RecyclerView.ViewHolder {
    private final MutableLiveData<Post> post = new MutableLiveData<>();

    public PostViewHolder(View itemView) {
        super(itemView);

        LiveData<String> name = Transformations.switchMap(post, new Function<Post, LiveData<String>>() {
            @Override
            public LiveData<String> apply(Post input) {
                return input.getLiveName();
            }
        });

        name.observeForever(new Observer<String>() {
            @Override
            public void onChanged(@Nullable String name) {
                // use name
            }
        });
    }

    public void bind(Post post) {
        post.setValue(post);
    }
}
Share:
36,547
Andrew Matiuk
Author by

Andrew Matiuk

Updated on July 16, 2022

Comments

  • Andrew Matiuk
    Andrew Matiuk almost 2 years

    There are many examples how to push new list to adapter on LiveData change.

    I'm trying to update one row (e.g number of comments for post) in the huge list. It would be stupid to reset whole list to change only one field.

    I am able to add observer onBindViewHolder, but I can't understand when should I remove observer

    @Override
    public void onBindViewHolder(ViewHolder vh, int position) {
        Post post = getPost(position);
        vh.itemView.setTag(post);
        post.getLiveName().observeForever(vh.nameObserver);
        ... 
    }