java - 聊天 fragment 不更新 recyclerview

标签 java android android-fragments android-recyclerview

我有 Fragment,其中有 recyclerview,它在第一次启动时工作正常(当 fragment 第一次启动时)。我可以发送和接收消息。但是当我从通知转到聊天 fragment 时,它停止更新 recyclerview。我必须从最近的应用程序中清除应用程序,然后当我从通知中转到聊天 fragment 时,它就可以正常工作。

每当我在点击通知后收到消息作为通知时,它都会打开同一个聊天屏幕(首先加载所有以前的消息),当我单击发送按钮发送消息时,它会发送给其他用户,但不会更新我的回收 View 。

正常加载聊天 fragment

getFragmentManager().beginTransaction().add(R.id.Navigation_Drawer, chatFragment).commit();

通过 Asynctask 从通知加载 fragment

((Navigation_Drawer)context).getFragmentManager().beginTransaction().replace(R.id.Navigation_Drawer, chatFragment).commit();

ChatScreenFragment

    public class Chat_Screen_Fragment extends Fragment implements View.OnClickListener, ChildEventListener{

    public static final String TAG = "###CHAT SCREEN###";

    List<Chat_Wrapper> message = new ArrayList<>();


    Chat_Adapter adapter;
    RecyclerView recyclerView;
    LinearLayoutManager layoutManager;


    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) {
        View v=inflater.inflate(R.layout.chat_screen_main_fragment,container,false);
        setRetainInstance(true);
        // GET INTENT VALUES FROM USER PROFILE CLASS

        UserName_Intent = getArguments().getString("Get_Name");
        UserImage_Intent = getArguments().getString("Get_Image");
        UserPhone_Intent = getArguments().getString("Get_Phone");
        UserID_Intent = getArguments().getString("Get_ID");
        FirebaseToken_Intent = getArguments().getString("Get_Token"); //Firebase Token of other person
        Room_Name_Intent = getArguments().getString("Get_Other"); // Room Name of chat
        UserLastSeen_Intent=getArguments().getString("LastSeen");
        //Sender_FCMToken = Session.getFirebaseID();
        // RECYCLER VIEW
        recyclerView = v.findViewById(R.id.Chat_Screen_Message_List);
        layoutManager = new LinearLayoutManager(getActivity());
        layoutManager.setStackFromEnd(true);
        recyclerView.setHasFixedSize(true);
        recyclerView.setLayoutManager(layoutManager);
databaseReference = FirebaseDatabase.getInstance().getReference().child(Room_Name_Intent);
        databaseReference.addChildEventListener(this);

        adapter = new Chat_Adapter(getActivity(), message);
        recyclerView.setAdapter(adapter);
        // FETCH OLD MESSAGE FROM DATABASE
        chatDatabase();
        return v;
    }
    // FIREBASE REAL TIME DATABASE WHICH FETCH ALL MESSAGES (SYNC) FROM ONLINE DATABASE
    @Override
    public void onChildAdded(DataSnapshot dataSnapshot, String s) {
        append_chat_conversation(dataSnapshot);
    }

    @Override
    public void onChildChanged(DataSnapshot dataSnapshot, String s) {
        append_chat_conversation(dataSnapshot);
    }

    @Override
    public void onChildRemoved(DataSnapshot dataSnapshot) {

    }

    @Override
    public void onChildMoved(DataSnapshot dataSnapshot, String s) {

    }

    @Override
    public void onCancelled(DatabaseError databaseError) {

    }
    private synchronized void append_chat_conversation(DataSnapshot dataSnapshot) {
        iterator = dataSnapshot.getChildren().iterator();
        while (iterator.hasNext()) {
            // NOW GET ALL DATA FROM FIREBASE DATABASE AND SAVE IT INTO STRINGS THEN CHECK EACH BY ITS MESSAGE TYPE
            Chat_Msg = (String) ((DataSnapshot) iterator.next()).getValue();
            Chat_FROM = (String) ((DataSnapshot) iterator.next()).getValue();
            Chat_FCM_TO= (String) ((DataSnapshot) iterator.next()).getValue();

            Chat_Database tempChatDatabase = new Chat_Database(getActivity());
            boolean hasValue=tempChatDatabase.CheckValueExist(_ID);
            if (!hasValue) {
                Log.d(TAG,"Chat Message "+Chat_Msg);
                    long id=chat_database.Insert_Chat(Session.getUserID(),Room_Name_Intent, UserID_Intent, "Text", Chat_Msg, Chat_FROM, Chat_TO, Chat_TimeStamp, Chat_FCM_FROM, Chat_FCM_TO, Session.getPhoneNO(), UserPhone_Intent,Random_ID,UserImage_Intent,UserLastSeen_Intent,Chat_FROM_ID);

                    //Adding Chat Data Into Database
                    Log.d(TAG,"Database Entry ID "+id);
                    if (id==0){
                        Log.d(TAG,"Database Already Has Value Of This Random Id ");
                        return;
                    }
                    Chat_Wrapper chat_wrapper = new Chat_Wrapper(Chat_Msg, null, null, null, null, null, null, Chat_TimeStamp, User_Intent, UserImage_Intent, Chat_FROM, null,null,id);
                    message.add(chat_wrapper);
                  adapter.notifyDataSetChanged();
                Log.d(TAG, "FIREBASE STORAGE PHOTO-3-MESSAGE ARRAY SIZE " + message.size());
                recyclerView.post(new Runnable() {
                    @Override
                    public void run() {
                        Log.d(TAG, "Moving to Bottom");
                        recyclerView.smoothScrollToPosition(adapter.getItemCount());

                    }
                });

                }               

                }

            }

            Log.d(TAG, "MESSAGE ARRAY SIZE " + message.size());

            tempChatDatabase.isDatabaseClose();
        }

        adapter.notifyDataSetChanged();
        recyclerView.post(new Runnable() {
            @Override
            public void run() {
                Log.d(TAG, "Moving to Bottom");
                recyclerView.smoothScrollToPosition(message.size()-1);
                //recyclerView.smoothScrollToPosition(adapter.getItemCount());

            }
        });
    }
    private void chatDatabase(){
        //Database Init and Filling Adapter
        Log.d(TAG,"Chat Database Function");
        chat_database=new Chat_Database(getActivity());
        chatCursor=chat_database.getUserChat(UserID_Intent);
        boolean checkDB_Exist=functions.DatabaseExist(getActivity(),"CHAT_DATABASE.DB");
        boolean chatItemsCounts=chatCursor.getCount()>0;
        chatCursor.moveToFirst();
        Log.d(TAG,"Value At Chat Database "+ checkDB_Exist+" "+chatItemsCounts);
        if (checkDB_Exist && chatCursor.getCount()>0 && chatCursor.getString(chatCursor.getColumnIndex("RECEIVER_USER_ID")).equals(UserID_Intent)){

            Log.d(TAG,"Database Exist Chat Database");
            message.clear();
            chatCursor.moveToFirst();
            do {
                database_rowID=chatCursor.getInt(chatCursor.getColumnIndex("ID"));
                database_userID=chatCursor.getString(chatCursor.getColumnIndex("USER_ID"));
                database_RoomName =chatCursor.getString(chatCursor.getColumnIndex("ROOM_NAME"));
                database_ReceiverID=chatCursor.getString(chatCursor.getColumnIndex("RECEIVER_USER_ID"));
                database_MessageType=chatCursor.getString(chatCursor.getColumnIndex("MESSAGE_TYPE"));
                database_Message=chatCursor.getString(chatCursor.getColumnIndex("USER_MESSAGE"));
                database_MsgFrom=chatCursor.getString(chatCursor.getColumnIndex("SENDER_NAME"));
                database_MsgTo=chatCursor.getString(chatCursor.getColumnIndex("RECEIVER_NAME"));
                database_TimeStamp=chatCursor.getString(chatCursor.getColumnIndex("TIME_STAMP"));
                database_FCMfrom=chatCursor.getString(chatCursor.getColumnIndex("SENDER_TOKEN"));
                database_FCMto=chatCursor.getString(chatCursor.getColumnIndex("RECEIVER_TOKEN"));
                database_LocalPath=chatCursor.getString(chatCursor.getColumnIndex("DOWNLOADED_AT"));
                database_PhoneFrom=chatCursor.getString(chatCursor.getColumnIndex("MY_PHONE"));
                database_PhoneTo=chatCursor.getString(chatCursor.getColumnIndex("OTHER_PHONE"));

                Log.d(TAG,"Value Of Database Message String = "+database_Message);
                Log.d(TAG,"Row ID of Database "+database_rowID);
                // Check Message Type


                    Log.d(TAG,"Message Type Is Text");
                    Chat_Wrapper text = new Chat_Wrapper(database_Message, null, null, null, null, null, null, database_TimeStamp, database_PhoneTo, UserImage_Intent, database_MsgFrom,null,null,database_rowID);
                    message.add(text);



            }
            while(chatCursor.moveToNext());

            Room_Name_Intent = database_RoomName;
            layoutManager = new LinearLayoutManager(getActivity());
            recyclerView.setLayoutManager(layoutManager);
            adapter.notifyDataSetChanged();
            chatCursor.close();
            boolean value = chat_database.isDatabaseClose();
            recyclerView.post(new Runnable() {
                @Override
                public void run() {
                    Log.d(TAG, "Moving to Bottom");
                    recyclerView.smoothScrollToPosition(message.size()-1);

                }
            });
            Log.d(TAG,"Value Of Database Close or Not "+value);

        }
    }
}

我还注意到当我评论 chatDatabase() 函数时从本地数据库获取早期消息的方法工作正常。但仍然无法解决问题。

万一工作完美。

  1. 当我从最近的应用程序中清除应用程序时
  2. 注释chatDatabase()函数
  3. 当我第一次启动 fragment 时。

最佳答案

虽然 Fragment 实例本身被保留,但每当分离/替换时,它都会销毁并重新创建其 View 。

将 Firebase 和适配器初始化逻辑移至 onCreate,仅在 onCreateView 中保留 RecyclerView 初始化。

您还应该重写 onDestroyView 并清除对 layoutManagerrecyclerView 的引用。

请注意,在这种情况下,您必须分离 Firebase 回调或处理 getView()(或 recyclerView,如果在 onDestroyView 中清除了引用)为 null 的情况,然后跳过更新 UI。

关于java - 聊天 fragment 不更新 recyclerview,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50814887/

相关文章:

java - 通知和通知所有显示等待多个线程的奇怪行为

java - fragment 不显示在线性布局的框架布局中

java - 如何添加按钮 Google+ 登录?

java - OData 服务器 API 和表达式语法、文字和成员

Android SpannableString 在部分文本后面设置背景

android - 参数 ITEMS 的非法修饰符;只允许 final

java - 有没有办法找出后栈发生了什么?

java - 如何为 fragment 设置列表适配器

java - 线程中的异常 "main"java.lang.NoClassDefFoundError : Could not initialize class org. springframework.jdbc.core.StatementCreatorUtils

android - tns-platform-declarations 的 nativescript-carousel 错误