如何在ChangeNotifier提供者Flutter中关闭函数

问题描述

如何在不处置的情况下关闭功能。我需要这个答案,因为当我注销时,需要关闭ChangeNotifier类中的函数

这是我的ChangeNotifier类:

class ChatandRequestProvider extends ChangeNotifier {
  bool _areThereNewChatsAndRequests = false;
  bool get areThereNewChatsAndRequests => _areThereNewChatsAndRequests;
  set areThereNewChatsAndRequests(bool value) {
    _areThereNewChatsAndRequests = value;
    notifyListeners();
  }

  List _chatsList = [];
  List get chatsList => _chatsList;

  set chatsList(List list) {
    _chatsList = list;
    notifyListeners();
  }

  getChats() async {
    var prefs = await SharedPreferences.getInstance();
    print('The getChats Id is ${prefs.getString(kUserId)}');
    FirebaseDatabase.instance
        .reference()
        .child('users')
        .child(prefs.getString(kUserId))
        .child('friendsArray')
        .onValue
        .listen((snapshot) {
      Map list = snapshot.snapshot.value;
      print('map is $list');
      var newItems = [];
      if (list != null) {
        list.forEach((key,value) {
          newItems.add(value);
        });
        chatsList = newItems;
        var globalArray = [];
        for (var item in newItems) {
          if (item[kLastTimestamp] != item[kLastTimestampSeen]) {
            areThereNewChatsAndRequests = true;
          }
          var status;
          switch (item['friendsstatus']) {
            case 'friends':
              status = RequestStatus.alreadyAFriend;
              break;
            case 'notFriends':
              status = RequestStatus.norequest;
              break;
            case 'blocked':
              status = RequestStatus.userThatBlockedMe;
              break;
            case 'unblocked':
              status = RequestStatus.norequest;
              break;
          }
          globalArray.add({kUserId: item[kUserId],kTypeOfRequest: status});
        }
        valuesList = globalArray;
      } else {
        deleteFromList(null,RequestStatus.alreadyAFriend);
        chatsList = [
          {kUserId: 'null'}
        ];
      }
    });
  }

例如,当我以user1身份登录并在LoadingScreen()中调用函数时,我得到了所有的朋友,而我可以转到聊天屏幕列表并与朋友聊天。到目前为止,还没有问题。但是,当我注销并使用另一个帐户登录时,假设说user2并再次调用函数,则由于两次调用函数,我将收到错误消息和两个响应。我没有使用Auth Packet,我在MongoDB上拥有自己的数据库,用于存储用户信息,但是请求和聊天记录则存储在RealTime数据库中。 所以我的问题是: 当user1注销我的应用程序时,我无法在provider上调用dispose(),因为如果他想再次登录到另一个帐户,由于Provider被处置,他将收到一个错误消息,因此当用户使用user1时我如何停止监听数据库注销并再次调用函数。非常感谢!!

解决方法

我不确定这是否行得通,因为我不完全了解您的应用程序流程,但您会这样说

我不能在提供者上调用dispose(),因为如果他想登录 再次转到另一个帐户,

用户注销时,应用程序是否应返回到配置提供程序的第一个屏幕? (除非您在MaterialApp中创建它,否则我也不确定)。您可以保存Firebase侦听器的实例,然后在注销/处置提供程序时将其关闭

var _myListener;

getChats() async {
    var prefs = await SharedPreferences.getInstance();
    print('The getChats Id is ${prefs.getString(kUserId)}');
    _myListener = FirebaseDatabase.instance
        .reference()
        .child('users')
        .child(prefs.getString(kUserId))
        .child('friendsArray')
        .onValue
        .listen((snapshot) ...
        ....
        /// The rest of your code
}

void closeListener(){ //call it when the user logs out
   _myListener?.close();
}

@override
void dispose(){
  closeListener(); // or call it in the dispose if you want 
                   //to dispose and create a new provider when the user logs out/ sign in
  super.dispose();
}