grid_bloc.dart 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. import 'dart:async';
  2. import 'package:dartz/dartz.dart';
  3. import 'package:flowy_sdk/log.dart';
  4. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-folder-data-model/view.pb.dart';
  6. import 'package:flowy_sdk/protobuf/flowy-grid-data-model/protobuf.dart';
  7. import 'package:flutter_bloc/flutter_bloc.dart';
  8. import 'package:freezed_annotation/freezed_annotation.dart';
  9. import 'field/grid_listenr.dart';
  10. import 'grid_listener.dart';
  11. import 'grid_service.dart';
  12. import 'row/row_service.dart';
  13. part 'grid_bloc.freezed.dart';
  14. class GridBloc extends Bloc<GridEvent, GridState> {
  15. final GridService _gridService;
  16. final GridListener _gridListener;
  17. final GridFieldsListener _fieldListener;
  18. GridBloc({required View view})
  19. : _fieldListener = GridFieldsListener(gridId: view.id),
  20. _gridService = GridService(gridId: view.id),
  21. _gridListener = GridListener(gridId: view.id),
  22. super(GridState.initial(view.id)) {
  23. on<GridEvent>(
  24. (event, emit) async {
  25. await event.map(
  26. initial: (InitialGrid value) async {
  27. await _initGrid(emit);
  28. _startListening();
  29. },
  30. createRow: (_CreateRow value) {
  31. _gridService.createRow();
  32. },
  33. didReceiveRowUpdate: (_DidReceiveRowUpdate value) {
  34. emit(state.copyWith(rows: value.rows, listState: value.listState));
  35. },
  36. didReceiveFieldUpdate: (_DidReceiveFieldUpdate value) {
  37. final rows = state.rows.map((row) => row.copyWith(fields: value.fields)).toList();
  38. emit(state.copyWith(
  39. rows: rows,
  40. fields: value.fields,
  41. listState: const GridListState.reload(),
  42. ));
  43. },
  44. );
  45. },
  46. );
  47. }
  48. @override
  49. Future<void> close() async {
  50. await _gridService.closeGrid();
  51. await _fieldListener.stop();
  52. await _gridListener.stop();
  53. return super.close();
  54. }
  55. Future<void> _initGrid(Emitter<GridState> emit) async {
  56. _fieldListener.updateFieldsNotifier.addPublishListener((result) {
  57. result.fold(
  58. (fields) => add(GridEvent.didReceiveFieldUpdate(fields)),
  59. (err) => Log.error(err),
  60. );
  61. });
  62. _fieldListener.start();
  63. await _loadGrid(emit);
  64. }
  65. void _startListening() {
  66. _gridListener.rowsUpdateNotifier.addPublishListener((result) {
  67. result.fold((gridBlockChangeset) {
  68. for (final changeset in gridBlockChangeset) {
  69. if (changeset.insertedRows.isNotEmpty) {
  70. _insertRows(changeset.insertedRows);
  71. }
  72. if (changeset.deletedRows.isNotEmpty) {
  73. _deleteRows(changeset.deletedRows);
  74. }
  75. if (changeset.updatedRows.isNotEmpty) {
  76. _updateRows(changeset.updatedRows);
  77. }
  78. }
  79. }, (err) => Log.error(err));
  80. });
  81. _gridListener.start();
  82. }
  83. Future<void> _loadGrid(Emitter<GridState> emit) async {
  84. final result = await _gridService.loadGrid();
  85. return Future(
  86. () => result.fold(
  87. (grid) async => await _loadFields(grid, emit),
  88. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  89. ),
  90. );
  91. }
  92. Future<void> _loadFields(Grid grid, Emitter<GridState> emit) async {
  93. final result = await _gridService.getFields(fieldOrders: grid.fieldOrders);
  94. return Future(
  95. () => result.fold(
  96. (fields) {
  97. emit(state.copyWith(
  98. grid: Some(grid),
  99. fields: fields.items,
  100. rows: _buildRows(grid.blockOrders, fields.items),
  101. loadingState: GridLoadingState.finish(left(unit)),
  102. ));
  103. },
  104. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  105. ),
  106. );
  107. }
  108. void _deleteRows(List<RowOrder> deletedRows) {
  109. final List<RowData> rows = [];
  110. final List<Tuple2<int, RowData>> deletedIndex = [];
  111. final Map<String, RowOrder> deletedRowMap = {for (var rowOrder in deletedRows) rowOrder.rowId: rowOrder};
  112. state.rows.asMap().forEach((index, value) {
  113. if (deletedRowMap[value.rowId] == null) {
  114. rows.add(value);
  115. } else {
  116. deletedIndex.add(Tuple2(index, value));
  117. }
  118. });
  119. add(GridEvent.didReceiveRowUpdate(rows, GridListState.delete(deletedIndex)));
  120. }
  121. void _insertRows(List<IndexRowOrder> createdRows) {
  122. final List<RowData> rows = List.from(state.rows);
  123. List<int> insertIndexs = [];
  124. for (final newRow in createdRows) {
  125. if (newRow.hasIndex()) {
  126. insertIndexs.add(newRow.index);
  127. rows.insert(newRow.index, _toRowData(newRow.rowOrder));
  128. } else {
  129. insertIndexs.add(rows.length);
  130. rows.add(_toRowData(newRow.rowOrder));
  131. }
  132. }
  133. add(GridEvent.didReceiveRowUpdate(rows, GridListState.insert(insertIndexs)));
  134. }
  135. void _updateRows(List<RowOrder> updatedRows) {
  136. final List<RowData> rows = List.from(state.rows);
  137. final List<int> updatedIndexs = [];
  138. for (final updatedRow in updatedRows) {
  139. final index = rows.indexWhere((row) => row.rowId == updatedRow.rowId);
  140. if (index != -1) {
  141. rows.removeAt(index);
  142. rows.insert(index, _toRowData(updatedRow));
  143. updatedIndexs.add(index);
  144. }
  145. }
  146. add(GridEvent.didReceiveRowUpdate(rows, const GridListState.reload()));
  147. }
  148. List<RowData> _buildRows(List<GridBlockOrder> blockOrders, List<Field> fields) {
  149. return blockOrders.expand((blockOrder) => blockOrder.rowOrders).map((rowOrder) {
  150. return RowData.fromBlockRow(state.gridId, rowOrder, fields);
  151. }).toList();
  152. }
  153. RowData _toRowData(RowOrder rowOrder) {
  154. return RowData.fromBlockRow(state.gridId, rowOrder, state.fields);
  155. }
  156. }
  157. @freezed
  158. class GridEvent with _$GridEvent {
  159. const factory GridEvent.initial() = InitialGrid;
  160. const factory GridEvent.createRow() = _CreateRow;
  161. const factory GridEvent.didReceiveRowUpdate(List<RowData> rows, GridListState listState) = _DidReceiveRowUpdate;
  162. const factory GridEvent.didReceiveFieldUpdate(List<Field> fields) = _DidReceiveFieldUpdate;
  163. }
  164. @freezed
  165. class GridState with _$GridState {
  166. const factory GridState({
  167. required String gridId,
  168. required Option<Grid> grid,
  169. required List<Field> fields,
  170. required List<RowData> rows,
  171. required GridLoadingState loadingState,
  172. required GridListState listState,
  173. }) = _GridState;
  174. factory GridState.initial(String gridId) => GridState(
  175. fields: [],
  176. rows: [],
  177. grid: none(),
  178. gridId: gridId,
  179. loadingState: const _Loading(),
  180. listState: const _Reload(),
  181. );
  182. }
  183. @freezed
  184. class GridLoadingState with _$GridLoadingState {
  185. const factory GridLoadingState.loading() = _Loading;
  186. const factory GridLoadingState.finish(Either<Unit, FlowyError> successOrFail) = _Finish;
  187. }
  188. @freezed
  189. class GridListState with _$GridListState {
  190. const factory GridListState.insert(List<int> indexs) = _Insert;
  191. const factory GridListState.delete(List<Tuple2<int, RowData>> indexs) = _Delete;
  192. const factory GridListState.reload() = _Reload;
  193. }