grid_bloc.dart 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. import 'dart:async';
  2. import 'package:dartz/dartz.dart';
  3. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  4. import 'package:flowy_sdk/protobuf/flowy-folder-data-model/view.pb.dart';
  5. import 'package:flowy_sdk/protobuf/flowy-grid-data-model/protobuf.dart';
  6. import 'package:flutter_bloc/flutter_bloc.dart';
  7. import 'package:freezed_annotation/freezed_annotation.dart';
  8. import 'grid_service.dart';
  9. import 'row/row_service.dart';
  10. part 'grid_bloc.freezed.dart';
  11. class GridBloc extends Bloc<GridEvent, GridState> {
  12. final GridService _gridService;
  13. final GridFieldCache fieldCache;
  14. late final GridRowCache rowCache;
  15. GridBloc({required View view})
  16. : _gridService = GridService(gridId: view.id),
  17. fieldCache = GridFieldCache(gridId: view.id),
  18. super(GridState.initial(view.id)) {
  19. rowCache = GridRowCache(
  20. gridId: view.id,
  21. dataDelegate: GridRowDataDelegateAdaptor(fieldCache),
  22. );
  23. on<GridEvent>(
  24. (event, emit) async {
  25. await event.map(
  26. initial: (InitialGrid value) async {
  27. _startListening();
  28. await _loadGrid(emit);
  29. },
  30. createRow: (_CreateRow value) {
  31. _gridService.createRow();
  32. },
  33. didReceiveRowUpdate: (_DidReceiveRowUpdate value) {
  34. emit(state.copyWith(rows: value.rows, listState: value.listState));
  35. },
  36. didReceiveFieldUpdate: (_DidReceiveFieldUpdate value) {
  37. emit(state.copyWith(rows: rowCache.clonedRows, fields: value.fields));
  38. },
  39. );
  40. },
  41. );
  42. }
  43. @override
  44. Future<void> close() async {
  45. await _gridService.closeGrid();
  46. await fieldCache.dispose();
  47. await rowCache.dispose();
  48. return super.close();
  49. }
  50. void _startListening() {
  51. fieldCache.addListener(
  52. listenWhen: () => !isClosed,
  53. onChanged: (fields) => add(GridEvent.didReceiveFieldUpdate(fields)),
  54. );
  55. rowCache.addListener(
  56. listenWhen: () => !isClosed,
  57. onChanged: (rows, listState) => add(GridEvent.didReceiveRowUpdate(rowCache.clonedRows, listState)),
  58. );
  59. }
  60. Future<void> _loadGrid(Emitter<GridState> emit) async {
  61. final result = await _gridService.loadGrid();
  62. return Future(
  63. () => result.fold(
  64. (grid) async => await _loadFields(grid, emit),
  65. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  66. ),
  67. );
  68. }
  69. Future<void> _loadFields(Grid grid, Emitter<GridState> emit) async {
  70. final result = await _gridService.getFields(fieldOrders: grid.fieldOrders);
  71. return Future(
  72. () => result.fold(
  73. (fields) {
  74. fieldCache.fields = fields.items;
  75. rowCache.updateWithBlock(grid.blockOrders);
  76. emit(state.copyWith(
  77. grid: Some(grid),
  78. fields: fieldCache.clonedFields,
  79. rows: rowCache.clonedRows,
  80. loadingState: GridLoadingState.finish(left(unit)),
  81. ));
  82. },
  83. (err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
  84. ),
  85. );
  86. }
  87. }
  88. @freezed
  89. class GridEvent with _$GridEvent {
  90. const factory GridEvent.initial() = InitialGrid;
  91. const factory GridEvent.createRow() = _CreateRow;
  92. const factory GridEvent.didReceiveRowUpdate(List<GridRow> rows, GridRowChangeReason listState) = _DidReceiveRowUpdate;
  93. const factory GridEvent.didReceiveFieldUpdate(List<Field> fields) = _DidReceiveFieldUpdate;
  94. }
  95. @freezed
  96. class GridState with _$GridState {
  97. const factory GridState({
  98. required String gridId,
  99. required Option<Grid> grid,
  100. required List<Field> fields,
  101. required List<GridRow> rows,
  102. required GridLoadingState loadingState,
  103. required GridRowChangeReason listState,
  104. }) = _GridState;
  105. factory GridState.initial(String gridId) => GridState(
  106. fields: [],
  107. rows: [],
  108. grid: none(),
  109. gridId: gridId,
  110. loadingState: const _Loading(),
  111. listState: const InitialListState(),
  112. );
  113. }
  114. @freezed
  115. class GridLoadingState with _$GridLoadingState {
  116. const factory GridLoadingState.loading() = _Loading;
  117. const factory GridLoadingState.finish(Either<Unit, FlowyError> successOrFail) = _Finish;
  118. }