board_bloc.dart 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. import 'dart:async';
  2. import 'package:app_flowy/plugins/grid/application/block/block_cache.dart';
  3. import 'package:app_flowy/plugins/grid/application/row/row_cache.dart';
  4. import 'package:appflowy_board/appflowy_board.dart';
  5. import 'package:dartz/dartz.dart';
  6. import 'package:equatable/equatable.dart';
  7. import 'package:flowy_sdk/log.dart';
  8. import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
  9. import 'package:flowy_sdk/protobuf/flowy-folder/view.pb.dart';
  10. import 'package:flowy_sdk/protobuf/flowy-grid/protobuf.dart';
  11. import 'package:flutter_bloc/flutter_bloc.dart';
  12. import 'package:freezed_annotation/freezed_annotation.dart';
  13. import 'dart:collection';
  14. import 'board_data_controller.dart';
  15. part 'board_bloc.freezed.dart';
  16. class BoardBloc extends Bloc<BoardEvent, BoardState> {
  17. final BoardDataController _dataController;
  18. late final AFBoardDataController boardDataController;
  19. BoardBloc({required ViewPB view})
  20. : _dataController = BoardDataController(view: view),
  21. super(BoardState.initial(view.id)) {
  22. boardDataController = AFBoardDataController(
  23. onMoveColumn: (
  24. fromIndex,
  25. toIndex,
  26. ) {},
  27. onMoveColumnItem: (
  28. columnId,
  29. fromIndex,
  30. toIndex,
  31. ) {},
  32. onMoveColumnItemToColumn: (
  33. fromColumnId,
  34. fromIndex,
  35. toColumnId,
  36. toIndex,
  37. ) {},
  38. );
  39. on<BoardEvent>(
  40. (event, emit) async {
  41. await event.when(
  42. initial: () async {
  43. _startListening();
  44. await _loadGrid(emit);
  45. },
  46. createRow: () {
  47. _dataController.createRow();
  48. },
  49. didReceiveGridUpdate: (GridPB grid) {
  50. emit(state.copyWith(grid: Some(grid)));
  51. },
  52. didReceiveGroups: (List<GroupPB> groups) {
  53. emit(state.copyWith(groups: groups));
  54. },
  55. );
  56. },
  57. );
  58. }
  59. @override
  60. Future<void> close() async {
  61. await _dataController.dispose();
  62. return super.close();
  63. }
  64. GridRowCache? getRowCache(String blockId, String rowId) {
  65. final GridBlockCache? blockCache = _dataController.blocks[blockId];
  66. return blockCache?.rowCache;
  67. }
  68. void _startListening() {
  69. _dataController.addListener(
  70. onGridChanged: (grid) {
  71. if (!isClosed) {
  72. add(BoardEvent.didReceiveGridUpdate(grid));
  73. }
  74. },
  75. onGroupChanged: (groups) {
  76. List<AFBoardColumnData> columns = groups.map((group) {
  77. return AFBoardColumnData(
  78. id: group.groupId,
  79. desc: group.desc,
  80. items: _buildRows(group.rows),
  81. customData: group,
  82. );
  83. }).toList();
  84. boardDataController.addColumns(columns);
  85. },
  86. onError: (err) {
  87. Log.error(err);
  88. },
  89. );
  90. }
  91. List<BoardColumnItem> _buildRows(List<RowPB> rows) {
  92. return rows.map((row) {
  93. final rowInfo = RowInfo(
  94. gridId: _dataController.gridId,
  95. blockId: row.blockId,
  96. id: row.id,
  97. fields: _dataController.fieldCache.unmodifiableFields,
  98. height: row.height.toDouble(),
  99. rawRow: row,
  100. );
  101. return BoardColumnItem(row: rowInfo);
  102. }).toList();
  103. }
  104. Future<void> _loadGrid(Emitter<BoardState> emit) async {
  105. final result = await _dataController.loadData();
  106. result.fold(
  107. (grid) => emit(
  108. state.copyWith(loadingState: GridLoadingState.finish(left(unit))),
  109. ),
  110. (err) => emit(
  111. state.copyWith(loadingState: GridLoadingState.finish(right(err))),
  112. ),
  113. );
  114. }
  115. }
  116. @freezed
  117. class BoardEvent with _$BoardEvent {
  118. const factory BoardEvent.initial() = InitialGrid;
  119. const factory BoardEvent.createRow() = _CreateRow;
  120. const factory BoardEvent.didReceiveGroups(List<GroupPB> groups) =
  121. _DidReceiveGroup;
  122. const factory BoardEvent.didReceiveGridUpdate(
  123. GridPB grid,
  124. ) = _DidReceiveGridUpdate;
  125. }
  126. @freezed
  127. class BoardState with _$BoardState {
  128. const factory BoardState({
  129. required String gridId,
  130. required Option<GridPB> grid,
  131. required List<GroupPB> groups,
  132. required List<RowInfo> rowInfos,
  133. required GridLoadingState loadingState,
  134. }) = _BoardState;
  135. factory BoardState.initial(String gridId) => BoardState(
  136. rowInfos: [],
  137. groups: [],
  138. grid: none(),
  139. gridId: gridId,
  140. loadingState: const _Loading(),
  141. );
  142. }
  143. @freezed
  144. class GridLoadingState with _$GridLoadingState {
  145. const factory GridLoadingState.loading() = _Loading;
  146. const factory GridLoadingState.finish(
  147. Either<Unit, FlowyError> successOrFail) = _Finish;
  148. }
  149. class GridFieldEquatable extends Equatable {
  150. final UnmodifiableListView<FieldPB> _fields;
  151. const GridFieldEquatable(
  152. UnmodifiableListView<FieldPB> fields,
  153. ) : _fields = fields;
  154. @override
  155. List<Object?> get props {
  156. if (_fields.isEmpty) {
  157. return [];
  158. }
  159. return [
  160. _fields.length,
  161. _fields
  162. .map((field) => field.width)
  163. .reduce((value, element) => value + element),
  164. ];
  165. }
  166. UnmodifiableListView<FieldPB> get value => UnmodifiableListView(_fields);
  167. }
  168. class BoardColumnItem extends AFColumnItem {
  169. final RowInfo row;
  170. BoardColumnItem({required this.row});
  171. @override
  172. String get id => row.id;
  173. }