board_bloc.dart 6.4 KB

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