board_bloc.dart 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. import 'dart:async';
  2. import 'package:app_flowy/plugins/grid/application/block/block_cache.dart';
  3. import 'package:app_flowy/plugins/grid/application/grid_data_controller.dart';
  4. import 'package:app_flowy/plugins/grid/application/row/row_cache.dart';
  5. import 'package:app_flowy/plugins/grid/presentation/widgets/header/type_option/builder.dart';
  6. import 'package:appflowy_board/appflowy_board.dart';
  7. import 'package:dartz/dartz.dart';
  8. import 'package:equatable/equatable.dart';
  9. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  10. import 'package:flowy_sdk/protobuf/flowy-folder/view.pb.dart';
  11. import 'package:flowy_sdk/protobuf/flowy-grid/protobuf.dart';
  12. import 'package:flutter_bloc/flutter_bloc.dart';
  13. import 'package:freezed_annotation/freezed_annotation.dart';
  14. import 'dart:collection';
  15. part 'board_bloc.freezed.dart';
  16. class BoardBloc extends Bloc<BoardEvent, BoardState> {
  17. final GridDataController _gridDataController;
  18. late final BoardDataController boardDataController;
  19. BoardBloc({required ViewPB view})
  20. : _gridDataController = GridDataController(view: view),
  21. super(BoardState.initial(view.id)) {
  22. boardDataController = BoardDataController(
  23. onMoveColumn: (
  24. fromIndex,
  25. toIndex,
  26. ) {},
  27. onMoveColumnItem: (
  28. columnId,
  29. fromIndex,
  30. toIndex,
  31. ) {},
  32. onMoveColumnItemToColumn: (
  33. fromColumnId,
  34. fromIndex,
  35. toColumnId,
  36. toIndex,
  37. ) {},
  38. );
  39. // boardDataController.addColumns(_buildColumns());
  40. on<BoardEvent>(
  41. (event, emit) async {
  42. await event.when(
  43. initial: () async {
  44. _startListening();
  45. await _loadGrid(emit);
  46. },
  47. createRow: () {
  48. _gridDataController.createRow();
  49. },
  50. didReceiveGridUpdate: (GridPB grid) {
  51. emit(state.copyWith(grid: Some(grid)));
  52. },
  53. didReceiveFieldUpdate: (UnmodifiableListView<GridFieldPB> fields) {
  54. emit(state.copyWith(fields: GridFieldEquatable(fields)));
  55. },
  56. didReceiveRowUpdate: (
  57. List<GridRowInfo> newRowInfos,
  58. GridRowChangeReason reason,
  59. ) {
  60. emit(state.copyWith(rowInfos: newRowInfos, reason: reason));
  61. },
  62. );
  63. },
  64. );
  65. }
  66. @override
  67. Future<void> close() async {
  68. await _gridDataController.dispose();
  69. return super.close();
  70. }
  71. GridRowCache? getRowCache(String blockId, String rowId) {
  72. final GridBlockCache? blockCache = _gridDataController.blocks[blockId];
  73. return blockCache?.rowCache;
  74. }
  75. void _startListening() {
  76. _gridDataController.addListener(
  77. onGridChanged: (grid) {
  78. if (!isClosed) {
  79. add(BoardEvent.didReceiveGridUpdate(grid));
  80. }
  81. },
  82. onRowsChanged: (rowInfos, reason) {
  83. if (!isClosed) {
  84. add(BoardEvent.didReceiveRowUpdate(rowInfos, reason));
  85. }
  86. },
  87. onFieldsChanged: (fields) {
  88. if (!isClosed) {
  89. _buildColumns(fields);
  90. add(BoardEvent.didReceiveFieldUpdate(fields));
  91. }
  92. },
  93. );
  94. }
  95. void _buildColumns(UnmodifiableListView<GridFieldPB> fields) {
  96. for (final field in fields) {
  97. if (field.fieldType == FieldType.SingleSelect) {
  98. _buildColumnsFromSingleSelect(field);
  99. }
  100. }
  101. }
  102. void _buildColumnsFromSingleSelect(GridFieldPB field) {
  103. final typeOptionContext = makeTypeOptionContext<SingleSelectTypeOptionPB>(
  104. gridId: _gridDataController.gridId,
  105. field: field,
  106. );
  107. typeOptionContext.loadTypeOptionData(
  108. onCompleted: (singleSelect) {
  109. List<BoardColumnData> columns = singleSelect.options.map((option) {
  110. return BoardColumnData(
  111. id: option.id,
  112. desc: option.name,
  113. customData: option,
  114. );
  115. }).toList();
  116. boardDataController.addColumns(columns);
  117. },
  118. onError: (err) {},
  119. );
  120. }
  121. Future<void> _loadGrid(Emitter<BoardState> emit) async {
  122. final result = await _gridDataController.loadData();
  123. result.fold(
  124. (grid) => emit(
  125. state.copyWith(loadingState: GridLoadingState.finish(left(unit))),
  126. ),
  127. (err) => emit(
  128. state.copyWith(loadingState: GridLoadingState.finish(right(err))),
  129. ),
  130. );
  131. }
  132. }
  133. @freezed
  134. class BoardEvent with _$BoardEvent {
  135. const factory BoardEvent.initial() = InitialGrid;
  136. const factory BoardEvent.createRow() = _CreateRow;
  137. const factory BoardEvent.didReceiveRowUpdate(
  138. List<GridRowInfo> rows,
  139. GridRowChangeReason listState,
  140. ) = _DidReceiveRowUpdate;
  141. const factory BoardEvent.didReceiveFieldUpdate(
  142. UnmodifiableListView<GridFieldPB> fields,
  143. ) = _DidReceiveFieldUpdate;
  144. const factory BoardEvent.didReceiveGridUpdate(
  145. GridPB grid,
  146. ) = _DidReceiveGridUpdate;
  147. }
  148. @freezed
  149. class BoardState with _$BoardState {
  150. const factory BoardState({
  151. required String gridId,
  152. required Option<GridPB> grid,
  153. required GridFieldEquatable fields,
  154. required List<GridRowInfo> rowInfos,
  155. required GridLoadingState loadingState,
  156. required GridRowChangeReason reason,
  157. }) = _BoardState;
  158. factory BoardState.initial(String gridId) => BoardState(
  159. fields: GridFieldEquatable(UnmodifiableListView([])),
  160. rowInfos: [],
  161. grid: none(),
  162. gridId: gridId,
  163. loadingState: const _Loading(),
  164. reason: const InitialListState(),
  165. );
  166. }
  167. @freezed
  168. class GridLoadingState with _$GridLoadingState {
  169. const factory GridLoadingState.loading() = _Loading;
  170. const factory GridLoadingState.finish(
  171. Either<Unit, FlowyError> successOrFail) = _Finish;
  172. }
  173. class GridFieldEquatable extends Equatable {
  174. final UnmodifiableListView<GridFieldPB> _fields;
  175. const GridFieldEquatable(
  176. UnmodifiableListView<GridFieldPB> fields,
  177. ) : _fields = fields;
  178. @override
  179. List<Object?> get props {
  180. if (_fields.isEmpty) {
  181. return [];
  182. }
  183. return [
  184. _fields.length,
  185. _fields
  186. .map((field) => field.width)
  187. .reduce((value, element) => value + element),
  188. ];
  189. }
  190. UnmodifiableListView<GridFieldPB> get value => UnmodifiableListView(_fields);
  191. }
  192. class TextItem extends ColumnItem {
  193. final String s;
  194. TextItem(this.s);
  195. @override
  196. String get id => s;
  197. }
  198. class RichTextItem extends ColumnItem {
  199. final String title;
  200. final String subtitle;
  201. RichTextItem({required this.title, required this.subtitle});
  202. @override
  203. String get id => title;
  204. }