board_bloc.dart 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  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:app_flowy/plugins/grid/application/row/row_service.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. import 'board_data_controller.dart';
  17. import 'group_controller.dart';
  18. part 'board_bloc.freezed.dart';
  19. class BoardBloc extends Bloc<BoardEvent, BoardState> {
  20. final BoardDataController _gridDataController;
  21. late final AFBoardDataController boardController;
  22. final MoveRowFFIService _rowService;
  23. LinkedHashMap<String, GroupController> groupControllers = LinkedHashMap.new();
  24. GridFieldCache get fieldCache => _gridDataController.fieldCache;
  25. String get gridId => _gridDataController.gridId;
  26. BoardBloc({required ViewPB view})
  27. : _rowService = MoveRowFFIService(gridId: view.id),
  28. _gridDataController = BoardDataController(view: view),
  29. super(BoardState.initial(view.id)) {
  30. boardController = AFBoardDataController(
  31. onMoveColumn: (
  32. fromColumnId,
  33. fromIndex,
  34. toColumnId,
  35. toIndex,
  36. ) {
  37. _moveGroup(fromColumnId, toColumnId);
  38. },
  39. onMoveColumnItem: (
  40. columnId,
  41. fromIndex,
  42. toIndex,
  43. ) {
  44. final fromRow = groupControllers[columnId]?.rowAtIndex(fromIndex);
  45. final toRow = groupControllers[columnId]?.rowAtIndex(toIndex);
  46. _moveRow(fromRow, columnId, toRow);
  47. },
  48. onMoveColumnItemToColumn: (
  49. fromColumnId,
  50. fromIndex,
  51. toColumnId,
  52. toIndex,
  53. ) {
  54. final fromRow = groupControllers[fromColumnId]?.rowAtIndex(fromIndex);
  55. final toRow = groupControllers[toColumnId]?.rowAtIndex(toIndex);
  56. _moveRow(fromRow, toColumnId, toRow);
  57. },
  58. );
  59. on<BoardEvent>(
  60. (event, emit) async {
  61. await event.when(
  62. initial: () async {
  63. _startListening();
  64. await _loadGrid(emit);
  65. },
  66. createRow: (groupId) async {
  67. final result = await _gridDataController.createBoardCard(groupId);
  68. result.fold(
  69. (_) {},
  70. (err) => Log.error(err),
  71. );
  72. },
  73. didCreateRow: (String groupId, RowPB row) {
  74. emit(state.copyWith(
  75. editingRow: Some(BoardEditingRow(columnId: groupId, row: row)),
  76. ));
  77. },
  78. endEditRow: (rowId) {
  79. assert(state.editingRow.isSome());
  80. state.editingRow.fold(() => null, (editingRow) {
  81. assert(editingRow.row.id == rowId);
  82. emit(state.copyWith(editingRow: none()));
  83. });
  84. },
  85. didReceiveGridUpdate: (GridPB grid) {
  86. emit(state.copyWith(grid: Some(grid)));
  87. },
  88. didReceiveError: (FlowyError error) {
  89. emit(state.copyWith(noneOrError: some(error)));
  90. },
  91. didReceiveGroups: (List<GroupPB> groups) {
  92. emit(state.copyWith(
  93. groupIds: groups.map((group) => group.groupId).toList(),
  94. ));
  95. },
  96. );
  97. },
  98. );
  99. }
  100. void _moveRow(RowPB? fromRow, String columnId, RowPB? toRow) {
  101. if (fromRow != null) {
  102. _rowService
  103. .moveGroupRow(
  104. fromRowId: fromRow.id,
  105. toGroupId: columnId,
  106. toRowId: toRow?.id,
  107. )
  108. .then((result) {
  109. result.fold((l) => null, (r) => add(BoardEvent.didReceiveError(r)));
  110. });
  111. }
  112. }
  113. void _moveGroup(String fromColumnId, String toColumnId) {
  114. _rowService
  115. .moveGroup(
  116. fromGroupId: fromColumnId,
  117. toGroupId: toColumnId,
  118. )
  119. .then((result) {
  120. result.fold((l) => null, (r) => add(BoardEvent.didReceiveError(r)));
  121. });
  122. }
  123. @override
  124. Future<void> close() async {
  125. await _gridDataController.dispose();
  126. for (final controller in groupControllers.values) {
  127. controller.dispose();
  128. }
  129. return super.close();
  130. }
  131. void initializeGroups(List<GroupPB> groups) {
  132. for (final group in groups) {
  133. final delegate = GroupControllerDelegateImpl(
  134. controller: boardController,
  135. didAddColumnItem: (groupId, row) {
  136. add(BoardEvent.didCreateRow(groupId, row));
  137. },
  138. );
  139. final controller = GroupController(
  140. gridId: state.gridId,
  141. group: group,
  142. delegate: delegate,
  143. );
  144. controller.startListening();
  145. groupControllers[controller.group.groupId] = (controller);
  146. }
  147. }
  148. GridRowCache? getRowCache(String blockId) {
  149. final GridBlockCache? blockCache = _gridDataController.blocks[blockId];
  150. return blockCache?.rowCache;
  151. }
  152. void _startListening() {
  153. _gridDataController.addListener(
  154. onGridChanged: (grid) {
  155. if (!isClosed) {
  156. add(BoardEvent.didReceiveGridUpdate(grid));
  157. }
  158. },
  159. didLoadGroups: (groups) {
  160. List<AFBoardColumnData> columns = groups.map((group) {
  161. return AFBoardColumnData(
  162. id: group.groupId,
  163. name: group.desc,
  164. items: _buildRows(group),
  165. customData: group,
  166. );
  167. }).toList();
  168. boardController.addColumns(columns);
  169. initializeGroups(groups);
  170. add(BoardEvent.didReceiveGroups(groups));
  171. },
  172. onDeletedGroup: (groupIds) {
  173. //
  174. },
  175. onInsertedGroup: (insertedGroups) {
  176. //
  177. },
  178. onUpdatedGroup: (updatedGroups) {
  179. //
  180. for (final group in updatedGroups) {
  181. final columnController =
  182. boardController.getColumnController(group.groupId);
  183. if (columnController != null) {
  184. columnController.updateColumnName(group.desc);
  185. }
  186. }
  187. },
  188. onError: (err) {
  189. Log.error(err);
  190. },
  191. );
  192. }
  193. List<AFColumnItem> _buildRows(GroupPB group) {
  194. final items = group.rows.map((row) {
  195. return BoardColumnItem(
  196. row: row,
  197. fieldId: group.fieldId,
  198. );
  199. }).toList();
  200. return <AFColumnItem>[...items];
  201. }
  202. Future<void> _loadGrid(Emitter<BoardState> emit) async {
  203. final result = await _gridDataController.loadData();
  204. result.fold(
  205. (grid) => emit(
  206. state.copyWith(loadingState: GridLoadingState.finish(left(unit))),
  207. ),
  208. (err) => emit(
  209. state.copyWith(loadingState: GridLoadingState.finish(right(err))),
  210. ),
  211. );
  212. }
  213. }
  214. @freezed
  215. class BoardEvent with _$BoardEvent {
  216. const factory BoardEvent.initial() = InitialBrid;
  217. const factory BoardEvent.createRow(String groupId) = _CreateRow;
  218. const factory BoardEvent.didCreateRow(String groupId, RowPB row) =
  219. _DidCreateRow;
  220. const factory BoardEvent.endEditRow(String rowId) = _EndEditRow;
  221. const factory BoardEvent.didReceiveError(FlowyError error) = _DidReceiveError;
  222. const factory BoardEvent.didReceiveGridUpdate(
  223. GridPB grid,
  224. ) = _DidReceiveGridUpdate;
  225. const factory BoardEvent.didReceiveGroups(List<GroupPB> groups) =
  226. _DidReceiveGroups;
  227. }
  228. @freezed
  229. class BoardState with _$BoardState {
  230. const factory BoardState({
  231. required String gridId,
  232. required Option<GridPB> grid,
  233. required List<String> groupIds,
  234. required Option<BoardEditingRow> editingRow,
  235. required GridLoadingState loadingState,
  236. required Option<FlowyError> noneOrError,
  237. }) = _BoardState;
  238. factory BoardState.initial(String gridId) => BoardState(
  239. grid: none(),
  240. gridId: gridId,
  241. groupIds: [],
  242. editingRow: none(),
  243. noneOrError: none(),
  244. loadingState: const _Loading(),
  245. );
  246. }
  247. @freezed
  248. class GridLoadingState with _$GridLoadingState {
  249. const factory GridLoadingState.loading() = _Loading;
  250. const factory GridLoadingState.finish(
  251. Either<Unit, FlowyError> successOrFail) = _Finish;
  252. }
  253. class GridFieldEquatable extends Equatable {
  254. final UnmodifiableListView<FieldPB> _fields;
  255. const GridFieldEquatable(
  256. UnmodifiableListView<FieldPB> fields,
  257. ) : _fields = fields;
  258. @override
  259. List<Object?> get props {
  260. if (_fields.isEmpty) {
  261. return [];
  262. }
  263. return [
  264. _fields.length,
  265. _fields
  266. .map((field) => field.width)
  267. .reduce((value, element) => value + element),
  268. ];
  269. }
  270. UnmodifiableListView<FieldPB> get value => UnmodifiableListView(_fields);
  271. }
  272. class BoardColumnItem extends AFColumnItem {
  273. final RowPB row;
  274. final String fieldId;
  275. final bool requestFocus;
  276. BoardColumnItem({
  277. required this.row,
  278. required this.fieldId,
  279. this.requestFocus = false,
  280. });
  281. @override
  282. String get id => row.id;
  283. }
  284. class GroupControllerDelegateImpl extends GroupControllerDelegate {
  285. final AFBoardDataController controller;
  286. final void Function(String, RowPB) didAddColumnItem;
  287. GroupControllerDelegateImpl({
  288. required this.controller,
  289. required this.didAddColumnItem,
  290. });
  291. @override
  292. void insertRow(GroupPB group, RowPB row, int? index) {
  293. if (index != null) {
  294. final item = BoardColumnItem(row: row, fieldId: group.fieldId);
  295. controller.insertColumnItem(group.groupId, index, item);
  296. } else {
  297. final item = BoardColumnItem(
  298. row: row,
  299. fieldId: group.fieldId,
  300. requestFocus: true,
  301. );
  302. controller.addColumnItem(group.groupId, item);
  303. didAddColumnItem(group.groupId, row);
  304. }
  305. }
  306. @override
  307. void removeRow(GroupPB group, String rowId) {
  308. controller.removeColumnItem(group.groupId, rowId);
  309. }
  310. @override
  311. void updateRow(GroupPB group, RowPB row) {
  312. controller.updateColumnItem(
  313. group.groupId,
  314. BoardColumnItem(
  315. row: row,
  316. fieldId: group.fieldId,
  317. ),
  318. );
  319. }
  320. }
  321. class BoardEditingRow {
  322. String columnId;
  323. RowPB row;
  324. BoardEditingRow({
  325. required this.columnId,
  326. required this.row,
  327. });
  328. }