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';
|
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-09 20:37:01 +08:00
|
|
|
part 'board_bloc.freezed.dart';
|
|
|
|
|
|
|
|
class BoardBloc extends Bloc<BoardEvent, BoardState> {
|
2022-08-11 15:00:36 +08:00
|
|
|
final BoardDataController _dataController;
|
2022-08-11 13:33:36 +08:00
|
|
|
late final AFBoardDataController boardDataController;
|
2022-08-09 20:37:01 +08:00
|
|
|
|
2022-08-12 20:10:56 +08:00
|
|
|
GridFieldCache get fieldCache => _dataController.fieldCache;
|
|
|
|
String get gridId => _dataController.gridId;
|
|
|
|
|
2022-08-09 20:37:01 +08:00
|
|
|
BoardBloc({required ViewPB view})
|
2022-08-11 15:00:36 +08:00
|
|
|
: _dataController = BoardDataController(view: view),
|
2022-08-09 20:37:01 +08:00
|
|
|
super(BoardState.initial(view.id)) {
|
2022-08-11 13:33:36 +08:00
|
|
|
boardDataController = AFBoardDataController(
|
2022-08-10 12:58:07 +08:00
|
|
|
onMoveColumn: (
|
|
|
|
fromIndex,
|
|
|
|
toIndex,
|
|
|
|
) {},
|
|
|
|
onMoveColumnItem: (
|
|
|
|
columnId,
|
|
|
|
fromIndex,
|
|
|
|
toIndex,
|
|
|
|
) {},
|
|
|
|
onMoveColumnItemToColumn: (
|
|
|
|
fromColumnId,
|
|
|
|
fromIndex,
|
|
|
|
toColumnId,
|
|
|
|
toIndex,
|
|
|
|
) {},
|
|
|
|
);
|
|
|
|
|
2022-08-09 20:37:01 +08:00
|
|
|
on<BoardEvent>(
|
|
|
|
(event, emit) async {
|
|
|
|
await event.when(
|
|
|
|
initial: () async {
|
|
|
|
_startListening();
|
|
|
|
await _loadGrid(emit);
|
|
|
|
},
|
|
|
|
createRow: () {
|
2022-08-11 15:00:36 +08:00
|
|
|
_dataController.createRow();
|
2022-08-09 20:37:01 +08:00
|
|
|
},
|
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)));
|
|
|
|
},
|
2022-08-11 21:18:27 +08:00
|
|
|
didReceiveGroups: (List<GroupPB> groups) {
|
|
|
|
emit(state.copyWith(groups: groups));
|
2022-08-10 20:12:05 +08:00
|
|
|
},
|
2022-08-12 20:10:56 +08:00
|
|
|
didReceiveRows: (List<RowInfo> rowInfos) {
|
|
|
|
emit(state.copyWith(rowInfos: rowInfos));
|
|
|
|
},
|
2022-08-09 20:37:01 +08:00
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<void> close() async {
|
2022-08-11 15:00:36 +08:00
|
|
|
await _dataController.dispose();
|
2022-08-09 20:37:01 +08:00
|
|
|
return super.close();
|
|
|
|
}
|
|
|
|
|
2022-08-12 20:10:56 +08:00
|
|
|
GridRowCache? getRowCache(String blockId) {
|
2022-08-11 15:00:36 +08:00
|
|
|
final GridBlockCache? blockCache = _dataController.blocks[blockId];
|
2022-08-09 20:37:01 +08:00
|
|
|
return blockCache?.rowCache;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _startListening() {
|
2022-08-11 15:00:36 +08:00
|
|
|
_dataController.addListener(
|
2022-08-09 20:37:01 +08:00
|
|
|
onGridChanged: (grid) {
|
|
|
|
if (!isClosed) {
|
|
|
|
add(BoardEvent.didReceiveGridUpdate(grid));
|
|
|
|
}
|
|
|
|
},
|
2022-08-11 21:18:27 +08:00
|
|
|
onGroupChanged: (groups) {
|
|
|
|
List<AFBoardColumnData> columns = groups.map((group) {
|
2022-08-11 13:33:36 +08:00
|
|
|
return AFBoardColumnData(
|
2022-08-11 21:18:27 +08:00
|
|
|
id: group.groupId,
|
|
|
|
desc: group.desc,
|
2022-08-12 16:06:30 +08:00
|
|
|
items: _buildRows(group.rows),
|
2022-08-11 21:18:27 +08:00
|
|
|
customData: group,
|
2022-08-10 16:03:41 +08:00
|
|
|
);
|
|
|
|
}).toList();
|
2022-08-10 12:58:07 +08:00
|
|
|
|
2022-08-10 16:03:41 +08:00
|
|
|
boardDataController.addColumns(columns);
|
|
|
|
},
|
2022-08-12 20:10:56 +08:00
|
|
|
onRowsChanged: (List<RowInfo> rowInfos, RowChangeReason reason) {
|
|
|
|
add(BoardEvent.didReceiveRows(rowInfos));
|
|
|
|
},
|
2022-08-11 21:18:27 +08:00
|
|
|
onError: (err) {
|
|
|
|
Log.error(err);
|
|
|
|
},
|
2022-08-10 16:03:41 +08:00
|
|
|
);
|
2022-08-10 12:58:07 +08:00
|
|
|
}
|
|
|
|
|
2022-08-12 16:06:30 +08:00
|
|
|
List<BoardColumnItem> _buildRows(List<RowPB> rows) {
|
|
|
|
return rows.map((row) {
|
2022-08-12 20:10:56 +08:00
|
|
|
// final rowInfo = RowInfo(
|
|
|
|
// gridId: _dataController.gridId,
|
|
|
|
// blockId: row.blockId,
|
|
|
|
// id: row.id,
|
|
|
|
// fields: _dataController.fieldCache.unmodifiableFields,
|
|
|
|
// height: row.height.toDouble(),
|
|
|
|
// rawRow: row,
|
|
|
|
// );
|
|
|
|
return BoardColumnItem(row: row);
|
2022-08-12 16:06:30 +08:00
|
|
|
}).toList();
|
|
|
|
}
|
|
|
|
|
2022-08-09 20:37:01 +08:00
|
|
|
Future<void> _loadGrid(Emitter<BoardState> emit) async {
|
2022-08-11 15:00:36 +08:00
|
|
|
final result = await _dataController.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 {
|
|
|
|
const factory BoardEvent.initial() = InitialGrid;
|
|
|
|
const factory BoardEvent.createRow() = _CreateRow;
|
2022-08-11 21:18:27 +08:00
|
|
|
const factory BoardEvent.didReceiveGroups(List<GroupPB> groups) =
|
|
|
|
_DidReceiveGroup;
|
2022-08-12 20:10:56 +08:00
|
|
|
const factory BoardEvent.didReceiveRows(List<RowInfo> rowInfos) =
|
|
|
|
_DidReceiveRows;
|
2022-08-09 20:37:01 +08:00
|
|
|
const factory BoardEvent.didReceiveGridUpdate(
|
|
|
|
GridPB grid,
|
|
|
|
) = _DidReceiveGridUpdate;
|
|
|
|
}
|
|
|
|
|
|
|
|
@freezed
|
|
|
|
class BoardState with _$BoardState {
|
|
|
|
const factory BoardState({
|
|
|
|
required String gridId,
|
|
|
|
required Option<GridPB> grid,
|
2022-08-11 21:18:27 +08:00
|
|
|
required List<GroupPB> groups,
|
2022-08-11 13:25:55 +08:00
|
|
|
required List<RowInfo> rowInfos,
|
2022-08-09 20:37:01 +08:00
|
|
|
required GridLoadingState loadingState,
|
|
|
|
}) = _BoardState;
|
|
|
|
|
|
|
|
factory BoardState.initial(String gridId) => BoardState(
|
|
|
|
rowInfos: [],
|
2022-08-11 21:18:27 +08:00
|
|
|
groups: [],
|
2022-08-09 20:37:01 +08:00
|
|
|
grid: none(),
|
|
|
|
gridId: gridId,
|
|
|
|
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 {
|
2022-08-11 13:25:55 +08:00
|
|
|
final UnmodifiableListView<FieldPB> _fields;
|
2022-08-09 20:37:01 +08:00
|
|
|
const GridFieldEquatable(
|
2022-08-11 13:25:55 +08:00
|
|
|
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),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2022-08-11 13:25:55 +08:00
|
|
|
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
|
|
|
|
2022-08-12 16:06:30 +08:00
|
|
|
BoardColumnItem({required this.row});
|
2022-08-10 12:58:07 +08:00
|
|
|
|
|
|
|
@override
|
2022-08-12 16:06:30 +08:00
|
|
|
String get id => row.id;
|
2022-08-10 12:58:07 +08:00
|
|
|
}
|