board_bloc.dart 11 KB

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