| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 | 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-data-model/view.pb.dart';import 'package:flowy_sdk/protobuf/flowy-grid-data-model/grid.pb.dart';import 'package:flutter/foundation.dart';import 'package:freezed_annotation/freezed_annotation.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 GridEventGetGridData(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;}class GridFieldCache {  final String gridId;  late final GridFieldsListener _fieldListener;  final FieldsNotifier _fieldNotifier = FieldsNotifier();  GridFieldCache({required this.gridId}) {    _fieldListener = GridFieldsListener(gridId: gridId);    _fieldListener.updateFieldsNotifier?.addPublishListener((result) {      result.fold(        (changeset) {          _deleteFields(changeset.deletedFields);          _insertFields(changeset.insertedFields);          _updateFields(changeset.updatedFields);        },        (err) => Log.error(err),      );    });    _fieldListener.start();  }  Future<void> dispose() async {    await _fieldListener.stop();    _fieldNotifier.dispose();  }  void applyChangeset(GridFieldChangeset changeset) {}  UnmodifiableListView<Field> get unmodifiableFields => UnmodifiableListView(_fieldNotifier.fields);  List<Field> get clonedFields => [..._fieldNotifier.fields];  set fields(List<Field> fields) {    _fieldNotifier.fields = [...fields];  }  VoidCallback addListener(      {VoidCallback? listener, void Function(List<Field>)? onChanged, bool Function()? listenWhen}) {    f() {      if (listenWhen != null && listenWhen() == false) {        return;      }      if (onChanged != null) {        onChanged(clonedFields);      }      if (listener != null) {        listener();      }    }    _fieldNotifier.addListener(f);    return f;  }  void removeListener(VoidCallback f) {    _fieldNotifier.removeListener(f);  }  void _deleteFields(List<FieldOrder> deletedFields) {    if (deletedFields.isEmpty) {      return;    }    final List<Field> fields = _fieldNotifier.fields;    final Map<String, FieldOrder> deletedFieldMap = {      for (var fieldOrder in deletedFields) fieldOrder.fieldId: fieldOrder    };    fields.retainWhere((field) => (deletedFieldMap[field.id] == null));    _fieldNotifier.fields = fields;  }  void _insertFields(List<IndexField> insertedFields) {    if (insertedFields.isEmpty) {      return;    }    final List<Field> fields = _fieldNotifier.fields;    for (final indexField in insertedFields) {      if (fields.length > indexField.index) {        fields.insert(indexField.index, indexField.field_1);      } else {        fields.add(indexField.field_1);      }    }    _fieldNotifier.fields = fields;  }  void _updateFields(List<Field> updatedFields) {    if (updatedFields.isEmpty) {      return;    }    final List<Field> fields = _fieldNotifier.fields;    for (final updatedField in updatedFields) {      final index = fields.indexWhere((field) => field.id == updatedField.id);      if (index != -1) {        fields.removeAt(index);        fields.insert(index, updatedField);      }    }    _fieldNotifier.fields = fields;  }}class GridRowDataDelegateAdaptor extends GridRowDataDelegate {  final GridFieldCache _cache;  GridRowDataDelegateAdaptor(GridFieldCache cache) : _cache = cache;  @override  UnmodifiableListView<Field> get fields => _cache.unmodifiableFields;  @override  GridRow buildGridRow(RowOrder rowOrder) {    return GridRow(      gridId: _cache.gridId,      fields: _cache.unmodifiableFields,      rowId: rowOrder.rowId,      height: rowOrder.height.toDouble(),    );  }  @override  void onFieldChanged(FieldDidUpdateCallback callback) {    _cache.addListener(listener: () {      callback();    });  }  @override  CellDataMap buildCellDataMap(Row rowData) {    var map = CellDataMap.new();    for (final field in fields) {      if (field.visibility) {        map[field.id] = GridCell(          rowId: rowData.id,          gridId: _cache.gridId,          cell: rowData.cellByFieldId[field.id],          field: field,        );      }    }    return map;  }}
 |