grid_bloc.dart 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  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-data-model/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. fieldDelegate: GridRowCacheDelegateImpl(fieldCache),
  30. );
  31. cellCache = GridCellCache(
  32. gridId: view.id,
  33. fieldDelegate: GridCellCacheDelegateImpl(fieldCache),
  34. );
  35. blockCache.start((result) {
  36. result.fold(
  37. (changesets) => rowCache.applyChangesets(changesets),
  38. (err) => Log.error(err),
  39. );
  40. });
  41. on<GridEvent>(
  42. (event, emit) async {
  43. await event.when(
  44. initial: () async {
  45. _startListening();
  46. await _loadGrid(emit);
  47. },
  48. createRow: () {
  49. _gridService.createRow();
  50. },
  51. didReceiveRowUpdate: (rows, listState) {
  52. emit(state.copyWith(rows: rows, listState: listState));
  53. },
  54. didReceiveFieldUpdate: (fields) {
  55. emit(state.copyWith(rows: rowCache.clonedRows, fields: GridFieldEquatable(fields)));
  56. },
  57. );
  58. },
  59. );
  60. }
  61. @override
  62. Future<void> close() async {
  63. await _gridService.closeGrid();
  64. await cellCache.dispose();
  65. await rowCache.dispose();
  66. await fieldCache.dispose();
  67. await blockCache.dispose();
  68. return super.close();
  69. }
  70. void _startListening() {
  71. fieldCache.addListener(
  72. listenWhen: () => !isClosed,
  73. onChanged: (fields) => add(GridEvent.didReceiveFieldUpdate(fields)),
  74. );
  75. rowCache.addListener(
  76. listenWhen: () => !isClosed,
  77. onChanged: (rows, listState) => add(GridEvent.didReceiveRowUpdate(rowCache.clonedRows, listState)),
  78. );
  79. }
  80. Future<void> _loadGrid(Emitter<GridState> emit) async {
  81. final result = await _gridService.loadGrid();
  82. return Future(
  83. () => result.fold(
  84. (grid) async {
  85. for (final block in grid.blocks) {
  86. blockCache.addBlockListener(block.id);
  87. }
  88. final rowOrders = grid.blocks.expand((block) => block.rowOrders).toList();
  89. rowCache.initialRows(rowOrders);
  90. await _loadFields(grid, emit);
  91. },
  92. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  93. ),
  94. );
  95. }
  96. Future<void> _loadFields(Grid grid, Emitter<GridState> emit) async {
  97. final result = await _gridService.getFields(fieldOrders: grid.fieldOrders);
  98. return Future(
  99. () => result.fold(
  100. (fields) {
  101. fieldCache.fields = fields.items;
  102. emit(state.copyWith(
  103. grid: Some(grid),
  104. fields: GridFieldEquatable(fieldCache.fields),
  105. rows: rowCache.clonedRows,
  106. loadingState: GridLoadingState.finish(left(unit)),
  107. ));
  108. },
  109. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  110. ),
  111. );
  112. }
  113. }
  114. @freezed
  115. class GridEvent with _$GridEvent {
  116. const factory GridEvent.initial() = InitialGrid;
  117. const factory GridEvent.createRow() = _CreateRow;
  118. const factory GridEvent.didReceiveRowUpdate(List<GridRow> rows, GridRowChangeReason listState) = _DidReceiveRowUpdate;
  119. const factory GridEvent.didReceiveFieldUpdate(List<Field> fields) = _DidReceiveFieldUpdate;
  120. }
  121. @freezed
  122. class GridState with _$GridState {
  123. const factory GridState({
  124. required String gridId,
  125. required Option<Grid> grid,
  126. required GridFieldEquatable fields,
  127. required List<GridRow> rows,
  128. required GridLoadingState loadingState,
  129. required GridRowChangeReason listState,
  130. }) = _GridState;
  131. factory GridState.initial(String gridId) => GridState(
  132. fields: const GridFieldEquatable([]),
  133. rows: [],
  134. grid: none(),
  135. gridId: gridId,
  136. loadingState: const _Loading(),
  137. listState: const InitialListState(),
  138. );
  139. }
  140. @freezed
  141. class GridLoadingState with _$GridLoadingState {
  142. const factory GridLoadingState.loading() = _Loading;
  143. const factory GridLoadingState.finish(Either<Unit, FlowyError> successOrFail) = _Finish;
  144. }
  145. class GridFieldEquatable extends Equatable {
  146. final List<Field> _fields;
  147. const GridFieldEquatable(List<Field> fields) : _fields = fields;
  148. @override
  149. List<Object?> get props {
  150. return [
  151. _fields.length,
  152. _fields.map((field) => field.width).reduce((value, element) => value + element),
  153. ];
  154. }
  155. UnmodifiableListView<Field> get value => UnmodifiableListView(_fields);
  156. }