grid_bloc.dart 5.1 KB

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