board_bloc.dart 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. import 'dart:async';
  2. import 'package:app_flowy/plugins/grid/application/block/block_cache.dart';
  3. import 'package:app_flowy/plugins/grid/application/field/field_cache.dart';
  4. import 'package:app_flowy/plugins/grid/application/row/row_cache.dart';
  5. import 'package:appflowy_board/appflowy_board.dart';
  6. import 'package:dartz/dartz.dart';
  7. import 'package:equatable/equatable.dart';
  8. import 'package:flowy_sdk/log.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. import 'board_data_controller.dart';
  16. import 'group_controller.dart';
  17. part 'board_bloc.freezed.dart';
  18. class BoardBloc extends Bloc<BoardEvent, BoardState> {
  19. final BoardDataController _dataController;
  20. late final AFBoardDataController afBoardDataController;
  21. List<GroupController> groupControllers = [];
  22. GridFieldCache get fieldCache => _dataController.fieldCache;
  23. String get gridId => _dataController.gridId;
  24. BoardBloc({required ViewPB view})
  25. : _dataController = BoardDataController(view: view),
  26. super(BoardState.initial(view.id)) {
  27. afBoardDataController = AFBoardDataController(
  28. onMoveColumn: (
  29. fromIndex,
  30. toIndex,
  31. ) {},
  32. onMoveColumnItem: (
  33. columnId,
  34. fromIndex,
  35. toIndex,
  36. ) {},
  37. onMoveColumnItemToColumn: (
  38. fromColumnId,
  39. fromIndex,
  40. toColumnId,
  41. toIndex,
  42. ) {},
  43. );
  44. on<BoardEvent>(
  45. (event, emit) async {
  46. await event.when(
  47. initial: () async {
  48. _startListening();
  49. await _loadGrid(emit);
  50. },
  51. createRow: (groupId) async {
  52. final result = await _dataController.createBoardCard(groupId);
  53. result.fold(
  54. (rowPB) {
  55. emit(state.copyWith(editingRow: some(rowPB)));
  56. },
  57. (err) => Log.error(err),
  58. );
  59. },
  60. endEditRow: (rowId) {
  61. assert(state.editingRow.isSome());
  62. state.editingRow.fold(() => null, (row) {
  63. assert(row.id == rowId);
  64. emit(state.copyWith(editingRow: none()));
  65. });
  66. },
  67. didReceiveGridUpdate: (GridPB grid) {
  68. emit(state.copyWith(grid: Some(grid)));
  69. },
  70. didReceiveRows: (List<RowInfo> rowInfos) {
  71. emit(state.copyWith(rowInfos: rowInfos));
  72. },
  73. );
  74. },
  75. );
  76. }
  77. @override
  78. Future<void> close() async {
  79. await _dataController.dispose();
  80. for (final controller in groupControllers) {
  81. controller.dispose();
  82. }
  83. return super.close();
  84. }
  85. void initializeGroups(List<GroupPB> groups) {
  86. for (final group in groups) {
  87. final delegate = GroupControllerDelegateImpl(afBoardDataController);
  88. final controller = GroupController(
  89. group: group,
  90. delegate: delegate,
  91. );
  92. controller.startListening();
  93. groupControllers.add(controller);
  94. }
  95. }
  96. GridRowCache? getRowCache(String blockId) {
  97. final GridBlockCache? blockCache = _dataController.blocks[blockId];
  98. return blockCache?.rowCache;
  99. }
  100. void _startListening() {
  101. _dataController.addListener(
  102. onGridChanged: (grid) {
  103. if (!isClosed) {
  104. add(BoardEvent.didReceiveGridUpdate(grid));
  105. }
  106. },
  107. didLoadGroups: (groups) {
  108. List<AFBoardColumnData> columns = groups.map((group) {
  109. return AFBoardColumnData(
  110. id: group.groupId,
  111. desc: group.desc,
  112. items: _buildRows(group.rows),
  113. customData: group,
  114. );
  115. }).toList();
  116. afBoardDataController.addColumns(columns);
  117. initializeGroups(groups);
  118. },
  119. onRowsChanged: (List<RowInfo> rowInfos, RowsChangedReason reason) {
  120. add(BoardEvent.didReceiveRows(rowInfos));
  121. },
  122. onError: (err) {
  123. Log.error(err);
  124. },
  125. );
  126. }
  127. List<AFColumnItem> _buildRows(List<RowPB> rows) {
  128. final items = rows.map((row) {
  129. return BoardColumnItem(row: row);
  130. }).toList();
  131. return <AFColumnItem>[...items];
  132. }
  133. Future<void> _loadGrid(Emitter<BoardState> emit) async {
  134. final result = await _dataController.loadData();
  135. result.fold(
  136. (grid) => emit(
  137. state.copyWith(loadingState: GridLoadingState.finish(left(unit))),
  138. ),
  139. (err) => emit(
  140. state.copyWith(loadingState: GridLoadingState.finish(right(err))),
  141. ),
  142. );
  143. }
  144. }
  145. @freezed
  146. class BoardEvent with _$BoardEvent {
  147. const factory BoardEvent.initial() = InitialGrid;
  148. const factory BoardEvent.createRow(String groupId) = _CreateRow;
  149. const factory BoardEvent.endEditRow(String rowId) = _EndEditRow;
  150. const factory BoardEvent.didReceiveRows(List<RowInfo> rowInfos) =
  151. _DidReceiveRows;
  152. const factory BoardEvent.didReceiveGridUpdate(
  153. GridPB grid,
  154. ) = _DidReceiveGridUpdate;
  155. }
  156. @freezed
  157. class BoardState with _$BoardState {
  158. const factory BoardState({
  159. required String gridId,
  160. required Option<GridPB> grid,
  161. required Option<RowPB> editingRow,
  162. required List<RowInfo> rowInfos,
  163. required GridLoadingState loadingState,
  164. }) = _BoardState;
  165. factory BoardState.initial(String gridId) => BoardState(
  166. rowInfos: [],
  167. grid: none(),
  168. gridId: gridId,
  169. editingRow: none(),
  170. loadingState: const _Loading(),
  171. );
  172. }
  173. @freezed
  174. class GridLoadingState with _$GridLoadingState {
  175. const factory GridLoadingState.loading() = _Loading;
  176. const factory GridLoadingState.finish(
  177. Either<Unit, FlowyError> successOrFail) = _Finish;
  178. }
  179. class GridFieldEquatable extends Equatable {
  180. final UnmodifiableListView<FieldPB> _fields;
  181. const GridFieldEquatable(
  182. UnmodifiableListView<FieldPB> fields,
  183. ) : _fields = fields;
  184. @override
  185. List<Object?> get props {
  186. if (_fields.isEmpty) {
  187. return [];
  188. }
  189. return [
  190. _fields.length,
  191. _fields
  192. .map((field) => field.width)
  193. .reduce((value, element) => value + element),
  194. ];
  195. }
  196. UnmodifiableListView<FieldPB> get value => UnmodifiableListView(_fields);
  197. }
  198. class BoardColumnItem extends AFColumnItem {
  199. final RowPB row;
  200. BoardColumnItem({required this.row});
  201. @override
  202. String get id => row.id;
  203. }
  204. class CreateCardItem extends AFColumnItem {
  205. @override
  206. String get id => '$CreateCardItem';
  207. }
  208. class GroupControllerDelegateImpl extends GroupControllerDelegate {
  209. final AFBoardDataController controller;
  210. GroupControllerDelegateImpl(this.controller);
  211. @override
  212. void insertRow(String groupId, RowPB row, int? index) {
  213. final item = BoardColumnItem(row: row);
  214. if (index != null) {
  215. controller.insertColumnItem(groupId, index, item);
  216. } else {
  217. controller.addColumnItem(groupId, item);
  218. }
  219. }
  220. @override
  221. void removeRow(String groupId, String rowId) {
  222. controller.removeColumnItem(groupId, rowId);
  223. }
  224. @override
  225. void updateRow(String groupId, RowPB row) {}
  226. }