Как транслировать событие из одного виджета с сохранением состояния в другой в Flutter

#android #flutter #kotlin #dart

#Android #Flutter #Kotlin #dart

Вопрос:

Я хотел бы транслировать событие из одного виджета с сохранением состояния в другой, но, похоже, не могу найти способ сделать это. Я установил этот плагин: event: ^ 1.1.4, но он не срабатывает. Я хочу что-то вроде приведенного ниже:

 Stateful Widget 1:
SomeEventClass.broadcastEvent();

Stateful Widget 2:
SomeEventClass.subscribe();
 

Комментарии:

1. хорошо, насколько я понял, вы хотите запустить функцию экрана A внутри экрана B. Правильно?

Ответ №1:

Используя что-то вроде этого.

 class ContinuousStream {
  _DataStore _mStorage = _DataStore.getInstance();

  /// Sets a new value [value] to the data stream [stream].
  /// If there are active subscribers, the value will be dispatched to them.
  void emit(String stream, var value) {
    _mStorage?.setValue(stream, value);
  }

  /// Subscribes to the given stream [stream].
  /// If stream already has data set, it will be delivered to the [callback] function.
  void on(String stream, void Function(Object) callback) {
    _mStorage?.setCallback(stream, callback);
  }

  /// Returns the current value of a given data [stream].
  Object getValue(String stream) {
    return _mStorage?.getValue(stream);
  }
}

// Storage class for ContinuousStream.
class _DataStore {
  // Singleton Instance for DataStore
  static _DataStore _instance;

  // Map instance to store data values with data stream.
  HashMap<String, _DataItem> _mDataItemsMap = HashMap();

  // Sets/Adds the new value to the given key.
  void setValue(String key, var value) {
    // Retrieve existing data item from map.
    _DataItem item = _mDataItemsMap[key];

    item ??= _DataItem();

    // Set new value to new/existing item.
    item.value = value;

    // Reset item to the map.
    _mDataItemsMap[key] = item;

    // Dispatch new value to all callbacks.
    item.callbacks?.forEach((callback) {
      callback(value);
    });
  }

  // Sets/Adds the new callback to the given data stream.
  void setCallback(String key, Function(Object) callback) {
    if (callback != null) {
      // Retrieve existing data item from the map.
      _DataItem item = _mDataItemsMap[key];

      item ??= _DataItem();

      // Retrieve callback functions from data item.
      List<Function(Object)> callbacks = item.callbacks;

      // Check if callback functions exists or not.
      if (callbacks == null) {
        // If it's null then create new List.
        callbacks = List();

        // Set callback functions list to data item.
        item.callbacks = callbacks;

        // Set the data item to the map.
        _mDataItemsMap[key] = item;
      }

      // Add the given callback into List of callback functions.
      callbacks.add(callback);

      // Dispatch value to the callback function if value already exists.
      if (item.value != null) {
        callback(item.value);
      }
    }
  }

  // Returns current value of the data stream.
  Object getValue(String key) {
    return _mDataItemsMap[key].value;
  }

  // Returns singleton instance of _DataStore
  static _DataStore getInstance() {
    _instance ??= _DataStore();
    return _instance;
  }
}

// Data class to hold value and callback functions of a data stream.
class _DataItem {
  var value;
  List<Function(Object)> callbacks;
}
 

После создания этого класса вы можете инициализировать поток и получить к нему доступ. Например, это необходимо сделать как в StatefulWidget 1, так и в StatefulWidget 2.

 ContinuousStream continuousStream = new ContinuousStream();
 

Затем из StateFulWidget1:

 void send() {
      String message = "Hello";
      continuousStream.emit("chat-message", message);
  }
 

Это соответствует вашему событию трансляции.

И из StatefulWidget2:

 continuousStream.on("chat-message", (message) {
      print("Message Received: $message");
    });
 

Это соответствует вашему событию подписки.
Это должно решить вашу проблему.

Комментарии:

1. Тонны ошибок типа A value of type 'Object?' can't be returned from the method 'getValue' because it has a return type of 'Object'