grid_service.dart 6.5 KB

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