grid_bloc.dart 4.4 KB

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