grid_service.dart 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. import 'package:dartz/dartz.dart';
  2. import 'package:flowy_sdk/dispatch/dispatch.dart';
  3. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  4. import 'package:flowy_sdk/protobuf/flowy-folder-data-model/view.pb.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-grid-data-model/grid.pb.dart';
  6. import 'package:flutter/foundation.dart';
  7. import 'package:freezed_annotation/freezed_annotation.dart';
  8. import 'package:app_flowy/workspace/application/grid/row/row_service.dart';
  9. part 'grid_service.freezed.dart';
  10. class GridService {
  11. final String gridId;
  12. GridService({
  13. required this.gridId,
  14. });
  15. Future<Either<Grid, FlowyError>> loadGrid() async {
  16. await FolderEventSetLatestView(ViewId(value: gridId)).send();
  17. final payload = GridId(value: gridId);
  18. return GridEventGetGridData(payload).send();
  19. }
  20. Future<Either<Row, FlowyError>> createRow({Option<String>? startRowId}) {
  21. CreateRowPayload payload = CreateRowPayload.create()..gridId = gridId;
  22. startRowId?.fold(() => null, (id) => payload.startRowId = id);
  23. return GridEventCreateRow(payload).send();
  24. }
  25. Future<Either<RepeatedField, FlowyError>> getFields({required List<FieldOrder> fieldOrders}) {
  26. final payload = QueryFieldPayload.create()
  27. ..gridId = gridId
  28. ..fieldOrders = RepeatedFieldOrder(items: fieldOrders);
  29. return GridEventGetFields(payload).send();
  30. }
  31. Future<Either<Unit, FlowyError>> closeGrid() {
  32. final request = ViewId(value: gridId);
  33. return FolderEventCloseView(request).send();
  34. }
  35. }
  36. class FieldsNotifier extends ChangeNotifier {
  37. List<Field> _fields = [];
  38. set fields(List<Field> fields) {
  39. _fields = fields;
  40. notifyListeners();
  41. }
  42. List<Field> get fields => _fields;
  43. }
  44. class GridFieldCache {
  45. final FieldsNotifier _fieldNotifier = FieldsNotifier();
  46. GridFieldCache();
  47. void applyChangeset(GridFieldChangeset changeset) {
  48. _removeFields(changeset.deletedFields);
  49. _insertFields(changeset.insertedFields);
  50. _updateFields(changeset.updatedFields);
  51. }
  52. UnmodifiableListView<Field> get unmodifiableFields => UnmodifiableListView(_fieldNotifier.fields);
  53. List<Field> get clonedFields => [..._fieldNotifier.fields];
  54. set clonedFields(List<Field> fields) {
  55. _fieldNotifier.fields = [...fields];
  56. }
  57. void listenOnFieldChanged(void Function(List<Field>) onFieldChanged) {
  58. _fieldNotifier.addListener(() => onFieldChanged(clonedFields));
  59. }
  60. void _removeFields(List<FieldOrder> deletedFields) {
  61. if (deletedFields.isEmpty) {
  62. return;
  63. }
  64. final List<Field> fields = _fieldNotifier.fields;
  65. final Map<String, FieldOrder> deletedFieldMap = {
  66. for (var fieldOrder in deletedFields) fieldOrder.fieldId: fieldOrder
  67. };
  68. fields.retainWhere((field) => (deletedFieldMap[field.id] == null));
  69. _fieldNotifier.fields = fields;
  70. }
  71. void _insertFields(List<IndexField> insertedFields) {
  72. if (insertedFields.isEmpty) {
  73. return;
  74. }
  75. final List<Field> fields = _fieldNotifier.fields;
  76. for (final indexField in insertedFields) {
  77. if (fields.length > indexField.index) {
  78. fields.removeAt(indexField.index);
  79. fields.insert(indexField.index, indexField.field_1);
  80. } else {
  81. fields.add(indexField.field_1);
  82. }
  83. }
  84. _fieldNotifier.fields = fields;
  85. }
  86. void _updateFields(List<Field> updatedFields) {
  87. if (updatedFields.isEmpty) {
  88. return;
  89. }
  90. final List<Field> fields = _fieldNotifier.fields;
  91. for (final updatedField in updatedFields) {
  92. final index = fields.indexWhere((field) => field.id == updatedField.id);
  93. if (index != -1) {
  94. fields.removeAt(index);
  95. fields.insert(index, updatedField);
  96. }
  97. }
  98. _fieldNotifier.fields = fields;
  99. }
  100. void dispose() {
  101. _fieldNotifier.dispose();
  102. }
  103. }
  104. class GridRowCache {
  105. final String gridId;
  106. UnmodifiableListView<Field> _fields = UnmodifiableListView([]);
  107. List<RowData> _rows = [];
  108. GridRowCache({required this.gridId});
  109. List<RowData> get rows => _rows;
  110. void updateWithBlock(List<GridBlockOrder> blocks, UnmodifiableListView<Field> fields) {
  111. _fields = fields;
  112. _rows = blocks.expand((block) => block.rowOrders).map((rowOrder) {
  113. return RowData.fromBlockRow(gridId, rowOrder, _fields);
  114. }).toList();
  115. }
  116. void updateFields(UnmodifiableListView<Field> fields) {
  117. if (fields.isEmpty) {
  118. return;
  119. }
  120. _fields = fields;
  121. _rows = _rows.map((row) => row.copyWith(fields: fields)).toList();
  122. }
  123. Option<GridListState> deleteRows(List<RowOrder> deletedRows) {
  124. if (deletedRows.isEmpty) {
  125. return none();
  126. }
  127. final List<RowData> newRows = [];
  128. final List<Tuple2<int, RowData>> deletedIndex = [];
  129. final Map<String, RowOrder> deletedRowMap = {for (var rowOrder in deletedRows) rowOrder.rowId: rowOrder};
  130. _rows.asMap().forEach((index, value) {
  131. if (deletedRowMap[value.rowId] == null) {
  132. newRows.add(value);
  133. } else {
  134. deletedIndex.add(Tuple2(index, value));
  135. }
  136. });
  137. _rows = newRows;
  138. return Some(GridListState.delete(deletedIndex));
  139. }
  140. Option<GridListState> insertRows(List<IndexRowOrder> createdRows) {
  141. if (createdRows.isEmpty) {
  142. return none();
  143. }
  144. List<int> insertIndexs = [];
  145. for (final newRow in createdRows) {
  146. if (newRow.hasIndex()) {
  147. insertIndexs.add(newRow.index);
  148. _rows.insert(newRow.index, _toRowData(newRow.rowOrder));
  149. } else {
  150. insertIndexs.add(_rows.length);
  151. _rows.add(_toRowData(newRow.rowOrder));
  152. }
  153. }
  154. return Some(GridListState.insert(insertIndexs));
  155. }
  156. void updateRows(List<RowOrder> updatedRows) {
  157. if (updatedRows.isEmpty) {
  158. return;
  159. }
  160. final List<int> updatedIndexs = [];
  161. for (final updatedRow in updatedRows) {
  162. final index = _rows.indexWhere((row) => row.rowId == updatedRow.rowId);
  163. if (index != -1) {
  164. _rows.removeAt(index);
  165. _rows.insert(index, _toRowData(updatedRow));
  166. updatedIndexs.add(index);
  167. }
  168. }
  169. }
  170. RowData _toRowData(RowOrder rowOrder) {
  171. return RowData.fromBlockRow(gridId, rowOrder, _fields);
  172. }
  173. }
  174. @freezed
  175. class GridListState with _$GridListState {
  176. const factory GridListState.insert(List<int> indexs) = _Insert;
  177. const factory GridListState.delete(List<Tuple2<int, RowData>> indexs) = _Delete;
  178. const factory GridListState.initial() = InitialListState;
  179. }