board_bloc.dart 5.9 KB

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