grid_service.dart 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  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 addListener(VoidCallback listener, {void Function(List<Field>)? onChanged}) {
  61. _fieldNotifier.addListener(() {
  62. if (onChanged != null) {
  63. onChanged(clonedFields);
  64. }
  65. listener();
  66. });
  67. }
  68. void _removeFields(List<FieldOrder> deletedFields) {
  69. if (deletedFields.isEmpty) {
  70. return;
  71. }
  72. final List<Field> fields = _fieldNotifier.fields;
  73. final Map<String, FieldOrder> deletedFieldMap = {
  74. for (var fieldOrder in deletedFields) fieldOrder.fieldId: fieldOrder
  75. };
  76. fields.retainWhere((field) => (deletedFieldMap[field.id] == null));
  77. _fieldNotifier.fields = fields;
  78. }
  79. void _insertFields(List<IndexField> insertedFields) {
  80. if (insertedFields.isEmpty) {
  81. return;
  82. }
  83. final List<Field> fields = _fieldNotifier.fields;
  84. for (final indexField in insertedFields) {
  85. if (fields.length > indexField.index) {
  86. fields.insert(indexField.index, indexField.field_1);
  87. } else {
  88. fields.add(indexField.field_1);
  89. }
  90. }
  91. _fieldNotifier.fields = fields;
  92. }
  93. void _updateFields(List<Field> updatedFields) {
  94. if (updatedFields.isEmpty) {
  95. return;
  96. }
  97. final List<Field> fields = _fieldNotifier.fields;
  98. for (final updatedField in updatedFields) {
  99. final index = fields.indexWhere((field) => field.id == updatedField.id);
  100. if (index != -1) {
  101. fields.removeAt(index);
  102. fields.insert(index, updatedField);
  103. }
  104. }
  105. _fieldNotifier.fields = fields;
  106. }
  107. void dispose() {
  108. _fieldNotifier.dispose();
  109. }
  110. }
  111. class GridRowCache {
  112. final String gridId;
  113. UnmodifiableListView<Field> _fields = UnmodifiableListView([]);
  114. List<RowData> _rows = [];
  115. GridRowCache({required this.gridId});
  116. List<RowData> get rows => [..._rows];
  117. void updateWithBlock(List<GridBlockOrder> blocks, UnmodifiableListView<Field> fields) {
  118. _fields = fields;
  119. _rows = blocks.expand((block) => block.rowOrders).map((rowOrder) {
  120. return RowData.fromBlockRow(gridId, rowOrder, _fields);
  121. }).toList();
  122. }
  123. void updateFields(UnmodifiableListView<Field> fields) {
  124. if (fields.isEmpty) {
  125. return;
  126. }
  127. _fields = fields;
  128. _rows = _rows.map((row) => row.copyWith(fields: fields)).toList();
  129. }
  130. Option<GridListState> deleteRows(List<RowOrder> deletedRows) {
  131. if (deletedRows.isEmpty) {
  132. return none();
  133. }
  134. final List<RowData> newRows = [];
  135. final DeletedIndex deletedIndex = [];
  136. final Map<String, RowOrder> deletedRowMap = {for (var rowOrder in deletedRows) rowOrder.rowId: rowOrder};
  137. _rows.asMap().forEach((index, value) {
  138. if (deletedRowMap[value.rowId] == null) {
  139. newRows.add(value);
  140. } else {
  141. deletedIndex.add(Tuple2(index, value));
  142. }
  143. });
  144. _rows = newRows;
  145. return Some(GridListState.delete(deletedIndex));
  146. }
  147. Option<GridListState> insertRows(List<IndexRowOrder> createdRows) {
  148. if (createdRows.isEmpty) {
  149. return none();
  150. }
  151. InsertedIndexs insertIndexs = [];
  152. for (final newRow in createdRows) {
  153. if (newRow.hasIndex()) {
  154. insertIndexs.add(Tuple2(newRow.index, newRow.rowOrder.rowId));
  155. _rows.insert(newRow.index, _toRowData(newRow.rowOrder));
  156. } else {
  157. insertIndexs.add(Tuple2(newRow.index, newRow.rowOrder.rowId));
  158. _rows.add(_toRowData(newRow.rowOrder));
  159. }
  160. }
  161. return Some(GridListState.insert(insertIndexs));
  162. }
  163. void updateRows(List<RowOrder> updatedRows) {
  164. if (updatedRows.isEmpty) {
  165. return;
  166. }
  167. final List<int> updatedIndexs = [];
  168. for (final updatedRow in updatedRows) {
  169. final index = _rows.indexWhere((row) => row.rowId == updatedRow.rowId);
  170. if (index != -1) {
  171. _rows.removeAt(index);
  172. _rows.insert(index, _toRowData(updatedRow));
  173. updatedIndexs.add(index);
  174. }
  175. }
  176. }
  177. RowData _toRowData(RowOrder rowOrder) {
  178. return RowData.fromBlockRow(gridId, rowOrder, _fields);
  179. }
  180. }
  181. typedef InsertedIndexs = List<Tuple2<int, String>>;
  182. typedef DeletedIndex = List<Tuple2<int, RowData>>;
  183. @freezed
  184. class GridListState with _$GridListState {
  185. const factory GridListState.insert(InsertedIndexs items) = _Insert;
  186. const factory GridListState.delete(DeletedIndex items) = _Delete;
  187. const factory GridListState.initial() = InitialListState;
  188. }