grid_bloc.dart 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. import 'dart:async';
  2. import 'package:dartz/dartz.dart';
  3. import 'package:equatable/equatable.dart';
  4. import 'package:appflowy_backend/protobuf/flowy-error/errors.pb.dart';
  5. import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
  6. import 'package:appflowy_backend/protobuf/flowy-database/protobuf.dart';
  7. import 'package:flutter_bloc/flutter_bloc.dart';
  8. import 'package:freezed_annotation/freezed_annotation.dart';
  9. import 'field/field_controller.dart';
  10. import 'grid_data_controller.dart';
  11. import 'row/row_cache.dart';
  12. import 'dart:collection';
  13. import 'row/row_service.dart';
  14. part 'grid_bloc.freezed.dart';
  15. class GridBloc extends Bloc<GridEvent, GridState> {
  16. final GridController gridController;
  17. void Function()? _createRowOperation;
  18. GridBloc({required ViewPB view, required this.gridController})
  19. : super(GridState.initial(view.id)) {
  20. on<GridEvent>(
  21. (event, emit) async {
  22. await event.when(
  23. initial: () async {
  24. _startListening();
  25. await _openGrid(emit);
  26. },
  27. createRow: () {
  28. state.loadingState.when(
  29. loading: () {
  30. _createRowOperation = () => gridController.createRow();
  31. },
  32. finish: (_) => gridController.createRow(),
  33. );
  34. },
  35. deleteRow: (rowInfo) async {
  36. final rowService = RowFFIService(
  37. databaseId: rowInfo.databaseId,
  38. );
  39. await rowService.deleteRow(rowInfo.rowPB.id);
  40. },
  41. didReceiveGridUpdate: (grid) {
  42. emit(state.copyWith(grid: Some(grid)));
  43. },
  44. didReceiveFieldUpdate: (fields) {
  45. emit(state.copyWith(
  46. fields: GridFieldEquatable(fields),
  47. ));
  48. },
  49. didReceiveRowUpdate: (newRowInfos, reason) {
  50. emit(state.copyWith(
  51. rowInfos: newRowInfos,
  52. rowCount: newRowInfos.length,
  53. reason: reason,
  54. ));
  55. },
  56. );
  57. },
  58. );
  59. }
  60. @override
  61. Future<void> close() async {
  62. await gridController.dispose();
  63. return super.close();
  64. }
  65. GridRowCache? getRowCache(String blockId, String rowId) {
  66. return gridController.rowCache;
  67. }
  68. void _startListening() {
  69. gridController.addListener(
  70. onGridChanged: (grid) {
  71. if (!isClosed) {
  72. add(GridEvent.didReceiveGridUpdate(grid));
  73. }
  74. },
  75. onRowsChanged: (rowInfos, reason) {
  76. if (!isClosed) {
  77. add(GridEvent.didReceiveRowUpdate(rowInfos, reason));
  78. }
  79. },
  80. onFieldsChanged: (fields) {
  81. if (!isClosed) {
  82. add(GridEvent.didReceiveFieldUpdate(fields));
  83. }
  84. },
  85. );
  86. }
  87. Future<void> _openGrid(Emitter<GridState> emit) async {
  88. final result = await gridController.openGrid();
  89. result.fold(
  90. (grid) {
  91. if (_createRowOperation != null) {
  92. _createRowOperation?.call();
  93. _createRowOperation = null;
  94. }
  95. emit(
  96. state.copyWith(loadingState: GridLoadingState.finish(left(unit))),
  97. );
  98. },
  99. (err) => emit(
  100. state.copyWith(loadingState: GridLoadingState.finish(right(err))),
  101. ),
  102. );
  103. }
  104. }
  105. @freezed
  106. class GridEvent with _$GridEvent {
  107. const factory GridEvent.initial() = InitialGrid;
  108. const factory GridEvent.createRow() = _CreateRow;
  109. const factory GridEvent.deleteRow(RowInfo rowInfo) = _DeleteRow;
  110. const factory GridEvent.didReceiveRowUpdate(
  111. List<RowInfo> rows,
  112. RowsChangedReason listState,
  113. ) = _DidReceiveRowUpdate;
  114. const factory GridEvent.didReceiveFieldUpdate(
  115. List<FieldInfo> fields,
  116. ) = _DidReceiveFieldUpdate;
  117. const factory GridEvent.didReceiveGridUpdate(
  118. DatabasePB grid,
  119. ) = _DidReceiveGridUpdate;
  120. }
  121. @freezed
  122. class GridState with _$GridState {
  123. const factory GridState({
  124. required String databaseId,
  125. required Option<DatabasePB> grid,
  126. required GridFieldEquatable fields,
  127. required List<RowInfo> rowInfos,
  128. required int rowCount,
  129. required GridLoadingState loadingState,
  130. required RowsChangedReason reason,
  131. }) = _GridState;
  132. factory GridState.initial(String databaseId) => GridState(
  133. fields: GridFieldEquatable(UnmodifiableListView([])),
  134. rowInfos: [],
  135. rowCount: 0,
  136. grid: none(),
  137. databaseId: databaseId,
  138. loadingState: const _Loading(),
  139. reason: const InitialListState(),
  140. );
  141. }
  142. @freezed
  143. class GridLoadingState with _$GridLoadingState {
  144. const factory GridLoadingState.loading() = _Loading;
  145. const factory GridLoadingState.finish(
  146. Either<Unit, FlowyError> successOrFail) = _Finish;
  147. }
  148. class GridFieldEquatable extends Equatable {
  149. final List<FieldInfo> _fields;
  150. const GridFieldEquatable(
  151. List<FieldInfo> fields,
  152. ) : _fields = fields;
  153. @override
  154. List<Object?> get props {
  155. if (_fields.isEmpty) {
  156. return [];
  157. }
  158. return [
  159. _fields.length,
  160. _fields
  161. .map((field) => field.width)
  162. .reduce((value, element) => value + element),
  163. ];
  164. }
  165. UnmodifiableListView<FieldInfo> get value => UnmodifiableListView(_fields);
  166. }