grid_bloc.dart 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. import 'dart:async';
  2. import 'package:dartz/dartz.dart';
  3. import 'package:equatable/equatable.dart';
  4. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-folder/view.pb.dart';
  6. import 'package:flowy_sdk/protobuf/flowy-grid/protobuf.dart';
  7. import 'package:flutter_bloc/flutter_bloc.dart';
  8. import 'package:freezed_annotation/freezed_annotation.dart';
  9. import 'block/block_cache.dart';
  10. import 'field/field_controller.dart';
  11. import 'grid_data_controller.dart';
  12. import 'row/row_cache.dart';
  13. import 'dart:collection';
  14. import 'row/row_service.dart';
  15. part 'grid_bloc.freezed.dart';
  16. class GridBloc extends Bloc<GridEvent, GridState> {
  17. final GridDataController dataController;
  18. void Function()? _createRowOperation;
  19. GridBloc({required ViewPB view})
  20. : dataController = GridDataController(view: view),
  21. super(GridState.initial(view.id)) {
  22. on<GridEvent>(
  23. (event, emit) async {
  24. await event.when(
  25. initial: () async {
  26. _startListening();
  27. await _loadGrid(emit);
  28. },
  29. createRow: () {
  30. state.loadingState.when(
  31. loading: () {
  32. _createRowOperation = () => dataController.createRow();
  33. },
  34. finish: (_) => dataController.createRow(),
  35. );
  36. },
  37. deleteRow: (rowInfo) async {
  38. final rowService = RowFFIService(
  39. blockId: rowInfo.rowPB.blockId,
  40. gridId: rowInfo.gridId,
  41. );
  42. await rowService.deleteRow(rowInfo.rowPB.id);
  43. },
  44. didReceiveGridUpdate: (grid) {
  45. emit(state.copyWith(grid: Some(grid)));
  46. },
  47. didReceiveFieldUpdate: (fields) {
  48. emit(state.copyWith(
  49. fields: GridFieldEquatable(fields),
  50. ));
  51. },
  52. didReceiveRowUpdate: (newRowInfos, reason) {
  53. emit(state.copyWith(
  54. rowInfos: newRowInfos,
  55. rowCount: newRowInfos.length,
  56. reason: reason,
  57. ));
  58. },
  59. );
  60. },
  61. );
  62. }
  63. @override
  64. Future<void> close() async {
  65. await dataController.dispose();
  66. return super.close();
  67. }
  68. GridRowCache? getRowCache(String blockId, String rowId) {
  69. final GridBlockCache? blockCache = dataController.blocks[blockId];
  70. return blockCache?.rowCache;
  71. }
  72. void _startListening() {
  73. dataController.addListener(
  74. onGridChanged: (grid) {
  75. if (!isClosed) {
  76. add(GridEvent.didReceiveGridUpdate(grid));
  77. }
  78. },
  79. onRowsChanged: (rowInfos, reason) {
  80. if (!isClosed) {
  81. add(GridEvent.didReceiveRowUpdate(rowInfos, reason));
  82. }
  83. },
  84. onFieldsChanged: (fields) {
  85. if (!isClosed) {
  86. add(GridEvent.didReceiveFieldUpdate(fields));
  87. }
  88. },
  89. );
  90. }
  91. Future<void> _loadGrid(Emitter<GridState> emit) async {
  92. final result = await dataController.loadData();
  93. result.fold(
  94. (grid) {
  95. if (_createRowOperation != null) {
  96. _createRowOperation?.call();
  97. _createRowOperation = null;
  98. }
  99. emit(
  100. state.copyWith(loadingState: GridLoadingState.finish(left(unit))),
  101. );
  102. },
  103. (err) => emit(
  104. state.copyWith(loadingState: GridLoadingState.finish(right(err))),
  105. ),
  106. );
  107. }
  108. }
  109. @freezed
  110. class GridEvent with _$GridEvent {
  111. const factory GridEvent.initial() = InitialGrid;
  112. const factory GridEvent.createRow() = _CreateRow;
  113. const factory GridEvent.deleteRow(RowInfo rowInfo) = _DeleteRow;
  114. const factory GridEvent.didReceiveRowUpdate(
  115. List<RowInfo> rows,
  116. RowsChangedReason listState,
  117. ) = _DidReceiveRowUpdate;
  118. const factory GridEvent.didReceiveFieldUpdate(
  119. UnmodifiableListView<GridFieldContext> fields,
  120. ) = _DidReceiveFieldUpdate;
  121. const factory GridEvent.didReceiveGridUpdate(
  122. GridPB grid,
  123. ) = _DidReceiveGridUpdate;
  124. }
  125. @freezed
  126. class GridState with _$GridState {
  127. const factory GridState({
  128. required String gridId,
  129. required Option<GridPB> grid,
  130. required GridFieldEquatable fields,
  131. required List<RowInfo> rowInfos,
  132. required int rowCount,
  133. required GridLoadingState loadingState,
  134. required RowsChangedReason reason,
  135. }) = _GridState;
  136. factory GridState.initial(String gridId) => GridState(
  137. fields: GridFieldEquatable(UnmodifiableListView([])),
  138. rowInfos: [],
  139. rowCount: 0,
  140. grid: none(),
  141. gridId: gridId,
  142. loadingState: const _Loading(),
  143. reason: const InitialListState(),
  144. );
  145. }
  146. @freezed
  147. class GridLoadingState with _$GridLoadingState {
  148. const factory GridLoadingState.loading() = _Loading;
  149. const factory GridLoadingState.finish(
  150. Either<Unit, FlowyError> successOrFail) = _Finish;
  151. }
  152. class GridFieldEquatable extends Equatable {
  153. final UnmodifiableListView<GridFieldContext> _fields;
  154. const GridFieldEquatable(
  155. UnmodifiableListView<GridFieldContext> fields,
  156. ) : _fields = fields;
  157. @override
  158. List<Object?> get props {
  159. if (_fields.isEmpty) {
  160. return [];
  161. }
  162. return [
  163. _fields.length,
  164. _fields
  165. .map((field) => field.width)
  166. .reduce((value, element) => value + element),
  167. ];
  168. }
  169. UnmodifiableListView<GridFieldContext> get value =>
  170. UnmodifiableListView(_fields);
  171. }