| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192 | import 'dart:collection';import 'package:app_flowy/plugins/grid/application/field/grid_listener.dart';import 'package:flowy_sdk/log.dart';import 'package:flowy_sdk/protobuf/flowy-grid/field_entities.pb.dart';import 'package:flutter/foundation.dart';import '../row/row_cache.dart';class FieldsNotifier extends ChangeNotifier {  List<FieldPB> _fields = [];  set fields(List<FieldPB> fields) {    _fields = fields;    notifyListeners();  }  List<FieldPB> get fields => _fields;}typedef FieldChangesetCallback = void Function(FieldChangesetPB);typedef FieldsCallback = void Function(List<FieldPB>);class GridFieldCache {  final String gridId;  final GridFieldsListener _fieldListener;  FieldsNotifier? _fieldNotifier = FieldsNotifier();  final Map<FieldsCallback, VoidCallback> _fieldsCallbackMap = {};  final Map<FieldChangesetCallback, FieldChangesetCallback>      _changesetCallbackMap = {};  GridFieldCache({required this.gridId})      : _fieldListener = GridFieldsListener(gridId: gridId) {    _fieldListener.start(onFieldsChanged: (result) {      result.fold(        (changeset) {          _deleteFields(changeset.deletedFields);          _insertFields(changeset.insertedFields);          _updateFields(changeset.updatedFields);          for (final listener in _changesetCallbackMap.values) {            listener(changeset);          }        },        (err) => Log.error(err),      );    });  }  Future<void> dispose() async {    await _fieldListener.stop();    _fieldNotifier?.dispose();    _fieldNotifier = null;  }  UnmodifiableListView<FieldPB> get unmodifiableFields =>      UnmodifiableListView(_fieldNotifier?.fields ?? []);  List<FieldPB> get fields => [..._fieldNotifier?.fields ?? []];  set fields(List<FieldPB> fields) {    _fieldNotifier?.fields = [...fields];  }  void addListener({    FieldsCallback? onFields,    FieldChangesetCallback? onChangeset,    bool Function()? listenWhen,  }) {    if (onChangeset != null) {      fn(c) {        if (listenWhen != null && listenWhen() == false) {          return;        }        onChangeset(c);      }      _changesetCallbackMap[onChangeset] = fn;    }    if (onFields != null) {      fn() {        if (listenWhen != null && listenWhen() == false) {          return;        }        onFields(fields);      }      _fieldsCallbackMap[onFields] = fn;      _fieldNotifier?.addListener(fn);    }  }  void removeListener({    FieldsCallback? onFieldsListener,    FieldChangesetCallback? onChangesetListener,  }) {    if (onFieldsListener != null) {      final fn = _fieldsCallbackMap.remove(onFieldsListener);      if (fn != null) {        _fieldNotifier?.removeListener(fn);      }    }    if (onChangesetListener != null) {      _changesetCallbackMap.remove(onChangesetListener);    }  }  void _deleteFields(List<FieldIdPB> deletedFields) {    if (deletedFields.isEmpty) {      return;    }    final List<FieldPB> newFields = fields;    final Map<String, FieldIdPB> deletedFieldMap = {      for (var fieldOrder in deletedFields) fieldOrder.fieldId: fieldOrder    };    newFields.retainWhere((field) => (deletedFieldMap[field.id] == null));    _fieldNotifier?.fields = newFields;  }  void _insertFields(List<IndexFieldPB> insertedFields) {    if (insertedFields.isEmpty) {      return;    }    final List<FieldPB> newFields = fields;    for (final indexField in insertedFields) {      if (newFields.length > indexField.index) {        newFields.insert(indexField.index, indexField.field_1);      } else {        newFields.add(indexField.field_1);      }    }    _fieldNotifier?.fields = newFields;  }  void _updateFields(List<FieldPB> updatedFields) {    if (updatedFields.isEmpty) {      return;    }    final List<FieldPB> newFields = fields;    for (final updatedField in updatedFields) {      final index =          newFields.indexWhere((field) => field.id == updatedField.id);      if (index != -1) {        newFields.removeAt(index);        newFields.insert(index, updatedField);      }    }    _fieldNotifier?.fields = newFields;  }}class GridRowFieldNotifierImpl extends IGridRowFieldNotifier {  final GridFieldCache _cache;  FieldChangesetCallback? _onChangesetFn;  FieldsCallback? _onFieldFn;  GridRowFieldNotifierImpl(GridFieldCache cache) : _cache = cache;  @override  UnmodifiableListView<FieldPB> get fields => _cache.unmodifiableFields;  @override  void onRowFieldsChanged(VoidCallback callback) {    _onFieldFn = (_) => callback();    _cache.addListener(onFields: _onFieldFn);  }  @override  void onRowFieldChanged(void Function(FieldPB) callback) {    _onChangesetFn = (FieldChangesetPB changeset) {      for (final updatedField in changeset.updatedFields) {        callback(updatedField);      }    };    _cache.addListener(onChangeset: _onChangesetFn);  }  @override  void onRowDispose() {    if (_onFieldFn != null) {      _cache.removeListener(onFieldsListener: _onFieldFn!);      _onFieldFn = null;    }    if (_onChangesetFn != null) {      _cache.removeListener(onChangesetListener: _onChangesetFn!);      _onChangesetFn = null;    }  }}
 |