| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 | import 'dart:collection';import 'package:app_flowy/workspace/application/grid/field/grid_listenr.dart';import 'package:dartz/dartz.dart';import 'package:flowy_sdk/dispatch/dispatch.dart';import 'package:flowy_sdk/log.dart';import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';import 'package:flowy_sdk/protobuf/flowy-folder/view.pb.dart';import 'package:flowy_sdk/protobuf/flowy-grid/block_entities.pb.dart';import 'package:flowy_sdk/protobuf/flowy-grid/field_entities.pb.dart';import 'package:flowy_sdk/protobuf/flowy-grid/grid_entities.pb.dart';import 'package:flowy_sdk/protobuf/flowy-grid/row_entities.pb.dart';import 'package:flutter/foundation.dart';import 'row/row_service.dart';class GridService {  final String gridId;  GridService({    required this.gridId,  });  Future<Either<Grid, FlowyError>> loadGrid() async {    await FolderEventSetLatestView(ViewId(value: gridId)).send();    final payload = GridId(value: gridId);    return GridEventGetGrid(payload).send();  }  Future<Either<Row, FlowyError>> createRow({Option<String>? startRowId}) {    CreateRowPayload payload = CreateRowPayload.create()..gridId = gridId;    startRowId?.fold(() => null, (id) => payload.startRowId = id);    return GridEventCreateRow(payload).send();  }  Future<Either<RepeatedField, FlowyError>> getFields({required List<FieldOrder> fieldOrders}) {    final payload = QueryFieldPayload.create()      ..gridId = gridId      ..fieldOrders = RepeatedFieldOrder(items: fieldOrders);    return GridEventGetFields(payload).send();  }  Future<Either<Unit, FlowyError>> closeGrid() {    final request = ViewId(value: gridId);    return FolderEventCloseView(request).send();  }}class FieldsNotifier extends ChangeNotifier {  List<Field> _fields = [];  set fields(List<Field> fields) {    _fields = fields;    notifyListeners();  }  List<Field> get fields => _fields;}typedef FieldChangesetCallback = void Function(GridFieldChangeset);typedef FieldsCallback = void Function(List<Field>);class GridFieldCache {  final String gridId;  late 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<Field> get unmodifiableFields => UnmodifiableListView(_fieldNotifier?.fields ?? []);  List<Field> get fields => [..._fieldNotifier?.fields ?? []];  set fields(List<Field> 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? onChangsetListener,  }) {    if (onFieldsListener != null) {      final fn = _fieldsCallbackMap.remove(onFieldsListener);      if (fn != null) {        _fieldNotifier?.removeListener(fn);      }    }    if (onChangsetListener != null) {      _changesetCallbackMap.remove(onChangsetListener);    }  }  void _deleteFields(List<FieldOrder> deletedFields) {    if (deletedFields.isEmpty) {      return;    }    final List<Field> newFields = fields;    final Map<String, FieldOrder> deletedFieldMap = {      for (var fieldOrder in deletedFields) fieldOrder.fieldId: fieldOrder    };    newFields.retainWhere((field) => (deletedFieldMap[field.id] == null));    _fieldNotifier?.fields = newFields;  }  void _insertFields(List<IndexField> insertedFields) {    if (insertedFields.isEmpty) {      return;    }    final List<Field> 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<Field> updatedFields) {    if (updatedFields.isEmpty) {      return;    }    final List<Field> 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 GridRowCacheDelegateImpl extends GridRowCacheDelegate {  final GridFieldCache _cache;  FieldChangesetCallback? _onChangesetFn;  FieldsCallback? _onFieldFn;  GridRowCacheDelegateImpl(GridFieldCache cache) : _cache = cache;  @override  UnmodifiableListView<Field> get fields => _cache.unmodifiableFields;  @override  void onFieldsChanged(VoidCallback callback) {    _onFieldFn = (_) => callback();    _cache.addListener(onFields: _onFieldFn);  }  @override  void onFieldUpdated(void Function(Field) callback) {    _onChangesetFn = (GridFieldChangeset changeset) {      for (final updatedField in changeset.updatedFields) {        callback(updatedField);      }    };    _cache.addListener(onChangeset: _onChangesetFn);  }  @override  void dispose() {    if (_onFieldFn != null) {      _cache.removeListener(onFieldsListener: _onFieldFn!);      _onFieldFn = null;    }    if (_onChangesetFn != null) {      _cache.removeListener(onChangsetListener: _onChangesetFn!);      _onChangesetFn = null;    }  }}
 |