400 lines
11 KiB
Dart
Raw Normal View History

2022-08-09 20:37:01 +08:00
import 'dart:async';
import 'package:app_flowy/plugins/grid/application/block/block_cache.dart';
2022-08-12 20:10:56 +08:00
import 'package:app_flowy/plugins/grid/application/field/field_cache.dart';
2022-08-09 20:37:01 +08:00
import 'package:app_flowy/plugins/grid/application/row/row_cache.dart';
import 'package:app_flowy/plugins/grid/application/row/row_service.dart';
2022-08-10 12:58:07 +08:00
import 'package:appflowy_board/appflowy_board.dart';
2022-08-09 20:37:01 +08:00
import 'package:dartz/dartz.dart';
import 'package:equatable/equatable.dart';
2022-08-10 17:59:28 +08:00
import 'package:flowy_sdk/log.dart';
2022-08-09 20:37:01 +08:00
import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
import 'package:flowy_sdk/protobuf/flowy-folder/view.pb.dart';
import 'package:flowy_sdk/protobuf/flowy-grid/protobuf.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:freezed_annotation/freezed_annotation.dart';
import 'dart:collection';
2022-08-11 15:00:36 +08:00
import 'board_data_controller.dart';
2022-08-16 17:13:56 +08:00
import 'group_controller.dart';
2022-08-11 15:00:36 +08:00
2022-08-09 20:37:01 +08:00
part 'board_bloc.freezed.dart';
class BoardBloc extends Bloc<BoardEvent, BoardState> {
2022-08-24 16:57:53 +08:00
final BoardDataController _gridDataController;
late final AFBoardDataController boardController;
final MoveRowFFIService _rowService;
2022-08-31 09:19:31 +08:00
LinkedHashMap<String, GroupController> groupControllers = LinkedHashMap();
2022-08-09 20:37:01 +08:00
2022-08-24 16:57:53 +08:00
GridFieldCache get fieldCache => _gridDataController.fieldCache;
String get gridId => _gridDataController.gridId;
2022-08-12 20:10:56 +08:00
2022-08-09 20:37:01 +08:00
BoardBloc({required ViewPB view})
: _rowService = MoveRowFFIService(gridId: view.id),
2022-08-24 16:57:53 +08:00
_gridDataController = BoardDataController(view: view),
2022-08-09 20:37:01 +08:00
super(BoardState.initial(view.id)) {
2022-08-24 16:57:53 +08:00
boardController = AFBoardDataController(
2022-08-10 12:58:07 +08:00
onMoveColumn: (
2022-08-22 16:16:15 +08:00
fromColumnId,
2022-08-10 12:58:07 +08:00
fromIndex,
2022-08-22 16:16:15 +08:00
toColumnId,
2022-08-10 12:58:07 +08:00
toIndex,
2022-08-22 16:16:15 +08:00
) {
_moveGroup(fromColumnId, toColumnId);
},
2022-08-10 12:58:07 +08:00
onMoveColumnItem: (
columnId,
fromIndex,
toIndex,
2022-08-17 19:29:14 +08:00
) {
final fromRow = groupControllers[columnId]?.rowAtIndex(fromIndex);
final toRow = groupControllers[columnId]?.rowAtIndex(toIndex);
2022-08-22 16:16:15 +08:00
_moveRow(fromRow, columnId, toRow);
2022-08-17 19:29:14 +08:00
},
2022-08-10 12:58:07 +08:00
onMoveColumnItemToColumn: (
fromColumnId,
fromIndex,
toColumnId,
toIndex,
2022-08-17 19:29:14 +08:00
) {
final fromRow = groupControllers[fromColumnId]?.rowAtIndex(fromIndex);
final toRow = groupControllers[toColumnId]?.rowAtIndex(toIndex);
2022-08-22 16:16:15 +08:00
_moveRow(fromRow, toColumnId, toRow);
2022-08-17 19:29:14 +08:00
},
2022-08-10 12:58:07 +08:00
);
2022-08-09 20:37:01 +08:00
on<BoardEvent>(
(event, emit) async {
await event.when(
initial: () async {
_startListening();
await _loadGrid(emit);
},
createBottomRow: (groupId) async {
final startRowId = groupControllers[groupId]?.lastRow()?.id;
final result = await _gridDataController.createBoardCard(
groupId,
startRowId: startRowId,
);
result.fold(
(_) {},
(err) => Log.error(err),
);
},
createHeaderRow: (String groupId) async {
2022-08-24 16:57:53 +08:00
final result = await _gridDataController.createBoardCard(groupId);
result.fold(
(_) {},
(err) => Log.error(err),
);
2022-08-09 20:37:01 +08:00
},
didCreateRow: (String groupId, RowPB row, int? index) {
emit(state.copyWith(
editingRow: Some(BoardEditingRow(
columnId: groupId,
row: row,
index: index,
)),
));
},
endEditRow: (rowId) {
assert(state.editingRow.isSome());
state.editingRow.fold(() => null, (editingRow) {
assert(editingRow.row.id == rowId);
emit(state.copyWith(editingRow: none()));
});
},
2022-08-10 12:58:07 +08:00
didReceiveGridUpdate: (GridPB grid) {
2022-08-09 20:37:01 +08:00
emit(state.copyWith(grid: Some(grid)));
},
didReceiveError: (FlowyError error) {
emit(state.copyWith(noneOrError: some(error)));
},
2022-08-26 11:51:42 +08:00
didReceiveGroups: (List<GroupPB> groups) {
emit(state.copyWith(
groupIds: groups.map((group) => group.groupId).toList(),
));
},
2022-08-09 20:37:01 +08:00
);
},
);
}
2022-08-22 16:16:15 +08:00
void _moveRow(RowPB? fromRow, String columnId, RowPB? toRow) {
if (fromRow != null) {
_rowService
2022-08-22 16:16:15 +08:00
.moveGroupRow(
fromRowId: fromRow.id,
2022-08-22 16:16:15 +08:00
toGroupId: columnId,
toRowId: toRow?.id,
)
.then((result) {
result.fold((l) => null, (r) => add(BoardEvent.didReceiveError(r)));
});
}
}
2022-08-22 16:16:15 +08:00
void _moveGroup(String fromColumnId, String toColumnId) {
_rowService
.moveGroup(
fromGroupId: fromColumnId,
toGroupId: toColumnId,
)
.then((result) {
result.fold((l) => null, (r) => add(BoardEvent.didReceiveError(r)));
});
}
2022-08-09 20:37:01 +08:00
@override
Future<void> close() async {
2022-08-24 16:57:53 +08:00
await _gridDataController.dispose();
2022-08-17 19:29:14 +08:00
for (final controller in groupControllers.values) {
2022-08-16 17:13:56 +08:00
controller.dispose();
}
2022-08-09 20:37:01 +08:00
return super.close();
}
2022-08-16 17:13:56 +08:00
void initializeGroups(List<GroupPB> groups) {
for (final group in groups) {
final delegate = GroupControllerDelegateImpl(
controller: boardController,
onNewColumnItem: (groupId, row, index) {
add(BoardEvent.didCreateRow(groupId, row, index));
},
);
2022-08-16 17:13:56 +08:00
final controller = GroupController(
2022-08-17 19:29:14 +08:00
gridId: state.gridId,
2022-08-16 17:13:56 +08:00
group: group,
delegate: delegate,
);
controller.startListening();
2022-08-17 19:29:14 +08:00
groupControllers[controller.group.groupId] = (controller);
2022-08-16 17:13:56 +08:00
}
}
2022-08-12 20:10:56 +08:00
GridRowCache? getRowCache(String blockId) {
2022-08-24 16:57:53 +08:00
final GridBlockCache? blockCache = _gridDataController.blocks[blockId];
2022-08-09 20:37:01 +08:00
return blockCache?.rowCache;
}
void _startListening() {
2022-08-24 16:57:53 +08:00
_gridDataController.addListener(
2022-08-09 20:37:01 +08:00
onGridChanged: (grid) {
if (!isClosed) {
add(BoardEvent.didReceiveGridUpdate(grid));
}
},
2022-08-16 17:13:56 +08:00
didLoadGroups: (groups) {
2022-08-11 21:18:27 +08:00
List<AFBoardColumnData> columns = groups.map((group) {
return AFBoardColumnData(
2022-08-11 21:18:27 +08:00
id: group.groupId,
2022-08-24 16:57:53 +08:00
name: group.desc,
items: _buildRows(group),
2022-08-11 21:18:27 +08:00
customData: group,
);
}).toList();
2022-08-10 12:58:07 +08:00
2022-08-24 16:57:53 +08:00
boardController.addColumns(columns);
2022-08-16 17:13:56 +08:00
initializeGroups(groups);
2022-08-26 11:51:42 +08:00
add(BoardEvent.didReceiveGroups(groups));
},
2022-08-24 16:57:53 +08:00
onDeletedGroup: (groupIds) {
//
},
onInsertedGroup: (insertedGroups) {
//
},
onUpdatedGroup: (updatedGroups) {
//
for (final group in updatedGroups) {
final columnController =
boardController.getColumnController(group.groupId);
if (columnController != null) {
columnController.updateColumnName(group.desc);
}
}
},
2022-08-11 21:18:27 +08:00
onError: (err) {
Log.error(err);
},
);
2022-08-10 12:58:07 +08:00
}
List<AFColumnItem> _buildRows(GroupPB group) {
final items = group.rows.map((row) {
return BoardColumnItem(
row: row,
fieldId: group.fieldId,
);
2022-08-12 16:06:30 +08:00
}).toList();
2022-08-13 11:51:26 +08:00
return <AFColumnItem>[...items];
2022-08-12 16:06:30 +08:00
}
2022-08-09 20:37:01 +08:00
Future<void> _loadGrid(Emitter<BoardState> emit) async {
2022-08-24 16:57:53 +08:00
final result = await _gridDataController.loadData();
2022-08-09 20:37:01 +08:00
result.fold(
(grid) => emit(
state.copyWith(loadingState: GridLoadingState.finish(left(unit))),
),
(err) => emit(
state.copyWith(loadingState: GridLoadingState.finish(right(err))),
),
);
}
}
@freezed
class BoardEvent with _$BoardEvent {
2022-08-29 10:06:29 +08:00
const factory BoardEvent.initial() = _InitialBoard;
const factory BoardEvent.createBottomRow(String groupId) = _CreateBottomRow;
const factory BoardEvent.createHeaderRow(String groupId) = _CreateHeaderRow;
const factory BoardEvent.didCreateRow(
String groupId,
RowPB row,
int? index,
) = _DidCreateRow;
const factory BoardEvent.endEditRow(String rowId) = _EndEditRow;
const factory BoardEvent.didReceiveError(FlowyError error) = _DidReceiveError;
2022-08-09 20:37:01 +08:00
const factory BoardEvent.didReceiveGridUpdate(
GridPB grid,
) = _DidReceiveGridUpdate;
2022-08-26 11:51:42 +08:00
const factory BoardEvent.didReceiveGroups(List<GroupPB> groups) =
_DidReceiveGroups;
2022-08-09 20:37:01 +08:00
}
@freezed
class BoardState with _$BoardState {
const factory BoardState({
required String gridId,
required Option<GridPB> grid,
2022-08-26 11:51:42 +08:00
required List<String> groupIds,
required Option<BoardEditingRow> editingRow,
2022-08-09 20:37:01 +08:00
required GridLoadingState loadingState,
required Option<FlowyError> noneOrError,
2022-08-09 20:37:01 +08:00
}) = _BoardState;
factory BoardState.initial(String gridId) => BoardState(
grid: none(),
gridId: gridId,
2022-08-26 11:51:42 +08:00
groupIds: [],
editingRow: none(),
noneOrError: none(),
2022-08-09 20:37:01 +08:00
loadingState: const _Loading(),
);
}
@freezed
class GridLoadingState with _$GridLoadingState {
const factory GridLoadingState.loading() = _Loading;
const factory GridLoadingState.finish(
Either<Unit, FlowyError> successOrFail) = _Finish;
}
class GridFieldEquatable extends Equatable {
final UnmodifiableListView<FieldPB> _fields;
2022-08-09 20:37:01 +08:00
const GridFieldEquatable(
UnmodifiableListView<FieldPB> fields,
2022-08-09 20:37:01 +08:00
) : _fields = fields;
@override
List<Object?> get props {
if (_fields.isEmpty) {
return [];
}
return [
_fields.length,
_fields
.map((field) => field.width)
.reduce((value, element) => value + element),
];
}
UnmodifiableListView<FieldPB> get value => UnmodifiableListView(_fields);
2022-08-09 20:37:01 +08:00
}
2022-08-10 12:58:07 +08:00
2022-08-12 16:06:30 +08:00
class BoardColumnItem extends AFColumnItem {
2022-08-12 20:10:56 +08:00
final RowPB row;
2022-08-10 12:58:07 +08:00
final String fieldId;
final bool requestFocus;
2022-08-10 12:58:07 +08:00
BoardColumnItem({
required this.row,
required this.fieldId,
this.requestFocus = false,
});
2022-08-13 11:57:14 +08:00
@override
String get id => row.id;
2022-08-13 11:57:14 +08:00
}
2022-08-16 17:13:56 +08:00
class GroupControllerDelegateImpl extends GroupControllerDelegate {
final AFBoardDataController controller;
final void Function(String, RowPB, int?) onNewColumnItem;
2022-08-16 17:13:56 +08:00
GroupControllerDelegateImpl({
required this.controller,
required this.onNewColumnItem,
});
2022-08-16 17:13:56 +08:00
@override
void insertRow(GroupPB group, RowPB row, int? index) {
2022-08-16 17:13:56 +08:00
if (index != null) {
final item = BoardColumnItem(row: row, fieldId: group.fieldId);
controller.insertColumnItem(group.groupId, index, item);
2022-08-16 17:13:56 +08:00
} else {
final item = BoardColumnItem(
row: row,
fieldId: group.fieldId,
);
controller.addColumnItem(group.groupId, item);
2022-08-16 17:13:56 +08:00
}
}
@override
void removeRow(GroupPB group, String rowId) {
controller.removeColumnItem(group.groupId, rowId);
2022-08-16 17:13:56 +08:00
}
@override
void updateRow(GroupPB group, RowPB row) {
controller.updateColumnItem(
group.groupId,
BoardColumnItem(
row: row,
fieldId: group.fieldId,
),
);
2022-08-16 20:34:12 +08:00
}
@override
void addNewRow(GroupPB group, RowPB row, int? index) {
final item = BoardColumnItem(
row: row,
fieldId: group.fieldId,
requestFocus: true,
);
if (index != null) {
controller.insertColumnItem(group.groupId, index, item);
} else {
controller.addColumnItem(group.groupId, item);
}
onNewColumnItem(group.groupId, row, index);
}
2022-08-16 17:13:56 +08:00
}
class BoardEditingRow {
String columnId;
RowPB row;
int? index;
BoardEditingRow({
required this.columnId,
required this.row,
required this.index,
});
}