Search code examples
javaandroidfirebasegoogle-cloud-firestorefirebaseui

How to set addSnapshotListener and remove in populateViewHolder in RecyclerView Item?


I have implemented RecyclerView with FirebaseUI-Android library.

My RecyclerView realtime data change well once i use that FirebaseRecyclerAdapter

In Collection that data document have field type as Boolean , Integer, Reference.

I want use that Reference to get data in populateViewHolder with addSnapshotListener.

Help me! Here is my code:

         FirebaseRecyclerAdapter<Conv, ConvViewHolder> firebaseConvAdapter = new FirebaseRecyclerAdapter<Conv, ConvViewHolder>(
            Conv.class,
            R.layout.users_single_layout,
            ConvViewHolder.class,
            conversationQuery
    ) {
        @Override
        protected void populateViewHolder(final ConvViewHolder convViewHolder, final Conv conv, int i) {

            final String list_user_id = getRef(i).getKey();

            final DocumentReference docRef = db.collection("cities").document(list_user_id);
            docRef.addSnapshotListener(new EventListener<DocumentSnapshot>() {
                @Override
                public void onEvent(@Nullable DocumentSnapshot snapshot,
                                    @Nullable FirebaseFirestoreException e) {
                    if (e != null) {
                        Log.w(TAG, "Listen failed.", e);
                        return;
                    }

                    if (snapshot != null && snapshot.exists()) {
                        Log.d(TAG, "Current data: " + snapshot.getData());
                    } else {
                        Log.d(TAG, "Current data: null");
                    }
                }
            });
        }
    };

    mConvList.setAdapter(firebaseConvAdapter);

Firebase saying if you add addSnapshotListener then must to remove it once no need for that Detach a listener

When you are no longer interested in listening to your data, you must detach your listener so that your event callbacks stop getting called. This allows the client to stop using bandwidth to receive updates. You can use the unsubscribe function on onSnapshot() to stop listening to updates.


Solution

  • To achieve this, you need to use a EventListener<DocumentSnapshot> like this:

    EventListener<DocumentSnapshot> eventListener = new EventListener<DocumentSnapshot>() {
        @Override
        public void onEvent(DocumentSnapshot snapshot, FirebaseFirestoreException e) {
            if (snapshot != null && snapshot.exists()) {
                //Do what you need to do
            }
        }
    };
    

    Declare a global ListenerRegistration listenerRegistration; variable and add the SnapshotListener in the place where is needed like this:

    if (listenerRegistration == null ) {
        listenerRegistration = yourRef.addSnapshotListener(eventListener);
    }
    

    To remove the listener, just use the following lines of code, in your onStop() method:

    @Override
    protected void onStop() {
        if (listenerRegistration != null) {
            listenerRegistration.remove();
        }
    }
    

    Also, don't forget to add it again once your onStart() method is called.

    @Override
    protected void onStart() {
        super.onStart();
        listenerRegistration = yourRef.addSnapshotListener(eventListener);
    }
    

    When you are calling addSnapshotListener for listening to realtime updates, it means that you attach a listener that gets called for every change that takes place in your database. So this is happening also when your app is closed, that's why it's mandatory to detach the listeners before the activity gets destroyed.

    If in your app you don't need to get the data in realtime, then you can simply use a get() call directly on the reference, which just reads the document only once. Since it only reads once, there is no listener to be removed. This is the correspondent of addListenerForSingleValueEvent() used in the Firebase realtime database.

    There is also a more elegant way for removing the listener which is to pass the activity as first the argument in the addSnapshotListener() method, so Firestore can clean up the listeners automatically when the activity is stopped.

    ListenerRegistration lg = yourDocumentRef
                .addSnapshotListener(YourActivity.this, eventListener);