grid_bloc.dart 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. import 'dart:async';
  2. import 'package:dartz/dartz.dart';
  3. import 'package:equatable/equatable.dart';
  4. import 'package:flowy_sdk/log.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  6. import 'package:flowy_sdk/protobuf/flowy-folder-data-model/view.pb.dart';
  7. import 'package:flowy_sdk/protobuf/flowy-grid/protobuf.dart';
  8. import 'package:flutter_bloc/flutter_bloc.dart';
  9. import 'package:freezed_annotation/freezed_annotation.dart';
  10. import 'block/block_listener.dart';
  11. import 'cell/cell_service/cell_service.dart';
  12. import 'grid_service.dart';
  13. import 'row/row_service.dart';
  14. import 'dart:collection';
  15. part 'grid_bloc.freezed.dart';
  16. class GridBloc extends Bloc<GridEvent, GridState> {
  17. final GridService _gridService;
  18. final GridFieldCache fieldCache;
  19. late final GridRowCache rowCache;
  20. late final GridCellCache cellCache;
  21. final GridBlockCache blockCache;
  22. GridBloc({required View view})
  23. : _gridService = GridService(gridId: view.id),
  24. fieldCache = GridFieldCache(gridId: view.id),
  25. blockCache = GridBlockCache(gridId: view.id),
  26. super(GridState.initial(view.id)) {
  27. rowCache = GridRowCache(
  28. gridId: view.id,
  29. blockId: "",
  30. fieldDelegate: GridRowCacheDelegateImpl(fieldCache),
  31. );
  32. cellCache = GridCellCache(
  33. gridId: view.id,
  34. fieldDelegate: GridCellCacheDelegateImpl(fieldCache),
  35. );
  36. blockCache.start((result) {
  37. result.fold(
  38. (changesets) => rowCache.applyChangesets(changesets),
  39. (err) => Log.error(err),
  40. );
  41. });
  42. on<GridEvent>(
  43. (event, emit) async {
  44. await event.when(
  45. initial: () async {
  46. _startListening();
  47. await _loadGrid(emit);
  48. },
  49. createRow: () {
  50. _gridService.createRow();
  51. },
  52. didReceiveRowUpdate: (rows, listState) {
  53. emit(state.copyWith(rows: rows, listState: listState));
  54. },
  55. didReceiveFieldUpdate: (fields) {
  56. emit(state.copyWith(rows: rowCache.clonedRows, fields: GridFieldEquatable(fields)));
  57. },
  58. );
  59. },
  60. );
  61. }
  62. @override
  63. Future<void> close() async {
  64. await _gridService.closeGrid();
  65. await cellCache.dispose();
  66. await rowCache.dispose();
  67. await fieldCache.dispose();
  68. await blockCache.dispose();
  69. return super.close();
  70. }
  71. void _startListening() {
  72. fieldCache.addListener(
  73. listenWhen: () => !isClosed,
  74. onChanged: (fields) => add(GridEvent.didReceiveFieldUpdate(fields)),
  75. );
  76. rowCache.addListener(
  77. listenWhen: () => !isClosed,
  78. onChanged: (rows, listState) => add(GridEvent.didReceiveRowUpdate(rowCache.clonedRows, listState)),
  79. );
  80. }
  81. Future<void> _loadGrid(Emitter<GridState> emit) async {
  82. final result = await _gridService.loadGrid();
  83. return Future(
  84. () => result.fold(
  85. (grid) async {
  86. for (final block in grid.blocks) {
  87. blockCache.addBlockListener(block.id);
  88. }
  89. final rowInfos = grid.blocks.expand((block) => block.rowInfos).toList();
  90. rowCache.initialRows(rowInfos);
  91. await _loadFields(grid, emit);
  92. },
  93. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  94. ),
  95. );
  96. }
  97. Future<void> _loadFields(Grid grid, Emitter<GridState> emit) async {
  98. final result = await _gridService.getFields(fieldOrders: grid.fieldOrders);
  99. return Future(
  100. () => result.fold(
  101. (fields) {
  102. fieldCache.fields = fields.items;
  103. emit(state.copyWith(
  104. grid: Some(grid),
  105. fields: GridFieldEquatable(fieldCache.fields),
  106. rows: rowCache.clonedRows,
  107. loadingState: GridLoadingState.finish(left(unit)),
  108. ));
  109. },
  110. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  111. ),
  112. );
  113. }
  114. }
  115. @freezed
  116. class GridEvent with _$GridEvent {
  117. const factory GridEvent.initial() = InitialGrid;
  118. const factory GridEvent.createRow() = _CreateRow;
  119. const factory GridEvent.didReceiveRowUpdate(List<GridRow> rows, GridRowChangeReason listState) = _DidReceiveRowUpdate;
  120. const factory GridEvent.didReceiveFieldUpdate(List<Field> fields) = _DidReceiveFieldUpdate;
  121. }
  122. @freezed
  123. class GridState with _$GridState {
  124. const factory GridState({
  125. required String gridId,
  126. required Option<Grid> grid,
  127. required GridFieldEquatable fields,
  128. required List<GridRow> rows,
  129. required GridLoadingState loadingState,
  130. required GridRowChangeReason listState,
  131. }) = _GridState;
  132. factory GridState.initial(String gridId) => GridState(
  133. fields: const GridFieldEquatable([]),
  134. rows: [],
  135. grid: none(),
  136. gridId: gridId,
  137. loadingState: const _Loading(),
  138. listState: const InitialListState(),
  139. );
  140. }
  141. @freezed
  142. class GridLoadingState with _$GridLoadingState {
  143. const factory GridLoadingState.loading() = _Loading;
  144. const factory GridLoadingState.finish(Either<Unit, FlowyError> successOrFail) = _Finish;
  145. }
  146. class GridFieldEquatable extends Equatable {
  147. final List<Field> _fields;
  148. const GridFieldEquatable(List<Field> fields) : _fields = fields;
  149. @override
  150. List<Object?> get props {
  151. return [
  152. _fields.length,
  153. _fields.map((field) => field.width).reduce((value, element) => value + element),
  154. ];
  155. }
  156. UnmodifiableListView<Field> get value => UnmodifiableListView(_fields);
  157. }