grid_bloc.dart 4.5 KB

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