grid_service.dart 6.1 KB

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