233 lines
6.3 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';
import 'package:app_flowy/plugins/grid/application/grid_data_controller.dart';
import 'package:app_flowy/plugins/grid/application/row/row_cache.dart';
import 'package:app_flowy/plugins/grid/presentation/widgets/header/type_option/builder.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';
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';
part 'board_bloc.freezed.dart';
class BoardBloc extends Bloc<BoardEvent, BoardState> {
2022-08-10 12:58:07 +08:00
final GridDataController _gridDataController;
late final BoardDataController boardDataController;
2022-08-09 20:37:01 +08:00
BoardBloc({required ViewPB view})
2022-08-10 12:58:07 +08:00
: _gridDataController = GridDataController(view: view),
2022-08-09 20:37:01 +08:00
super(BoardState.initial(view.id)) {
2022-08-10 12:58:07 +08:00
boardDataController = BoardDataController(
onMoveColumn: (
fromIndex,
toIndex,
) {},
onMoveColumnItem: (
columnId,
fromIndex,
toIndex,
) {},
onMoveColumnItemToColumn: (
fromColumnId,
fromIndex,
toColumnId,
toIndex,
) {},
);
// boardDataController.addColumns(_buildColumns());
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-10 12:58:07 +08:00
_gridDataController.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-10 12:58:07 +08:00
didReceiveFieldUpdate: (UnmodifiableListView<GridFieldPB> fields) {
emit(state.copyWith(fields: GridFieldEquatable(fields)));
2022-08-09 20:37:01 +08:00
},
2022-08-10 12:58:07 +08:00
didReceiveRowUpdate: (
List<GridRowInfo> newRowInfos,
GridRowChangeReason reason,
) {
emit(state.copyWith(rowInfos: newRowInfos, reason: reason));
2022-08-09 20:37:01 +08:00
},
);
},
);
}
@override
Future<void> close() async {
2022-08-10 12:58:07 +08:00
await _gridDataController.dispose();
2022-08-09 20:37:01 +08:00
return super.close();
}
GridRowCache? getRowCache(String blockId, String rowId) {
2022-08-10 12:58:07 +08:00
final GridBlockCache? blockCache = _gridDataController.blocks[blockId];
2022-08-09 20:37:01 +08:00
return blockCache?.rowCache;
}
void _startListening() {
2022-08-10 12:58:07 +08:00
_gridDataController.addListener(
2022-08-09 20:37:01 +08:00
onGridChanged: (grid) {
if (!isClosed) {
add(BoardEvent.didReceiveGridUpdate(grid));
}
},
onRowsChanged: (rowInfos, reason) {
if (!isClosed) {
add(BoardEvent.didReceiveRowUpdate(rowInfos, reason));
}
},
onFieldsChanged: (fields) {
if (!isClosed) {
2022-08-10 12:58:07 +08:00
_buildColumns(fields);
2022-08-09 20:37:01 +08:00
add(BoardEvent.didReceiveFieldUpdate(fields));
}
},
);
}
2022-08-10 12:58:07 +08:00
void _buildColumns(UnmodifiableListView<GridFieldPB> fields) {
for (final field in fields) {
if (field.fieldType == FieldType.SingleSelect) {
_buildColumnsFromSingleSelect(field);
2022-08-10 12:58:07 +08:00
}
}
}
2022-08-10 12:58:07 +08:00
void _buildColumnsFromSingleSelect(GridFieldPB field) {
final typeOptionContext = makeTypeOptionContext<SingleSelectTypeOptionPB>(
gridId: _gridDataController.gridId,
field: field,
);
2022-08-10 12:58:07 +08:00
typeOptionContext.loadTypeOptionData(
onCompleted: (singleSelect) {
List<BoardColumnData> columns = singleSelect.options.map((option) {
return BoardColumnData(
id: option.id,
desc: option.name,
customData: option,
);
}).toList();
2022-08-10 12:58:07 +08:00
boardDataController.addColumns(columns);
},
onError: (err) {},
);
2022-08-10 12:58:07 +08:00
}
2022-08-09 20:37:01 +08:00
Future<void> _loadGrid(Emitter<BoardState> emit) async {
2022-08-10 12:58:07 +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 {
const factory BoardEvent.initial() = InitialGrid;
const factory BoardEvent.createRow() = _CreateRow;
const factory BoardEvent.didReceiveRowUpdate(
List<GridRowInfo> rows,
GridRowChangeReason listState,
) = _DidReceiveRowUpdate;
const factory BoardEvent.didReceiveFieldUpdate(
UnmodifiableListView<GridFieldPB> fields,
) = _DidReceiveFieldUpdate;
const factory BoardEvent.didReceiveGridUpdate(
GridPB grid,
) = _DidReceiveGridUpdate;
}
@freezed
class BoardState with _$BoardState {
const factory BoardState({
required String gridId,
required Option<GridPB> grid,
required GridFieldEquatable fields,
required List<GridRowInfo> rowInfos,
required GridLoadingState loadingState,
required GridRowChangeReason reason,
}) = _BoardState;
factory BoardState.initial(String gridId) => BoardState(
fields: GridFieldEquatable(UnmodifiableListView([])),
rowInfos: [],
grid: none(),
gridId: gridId,
loadingState: const _Loading(),
reason: const InitialListState(),
);
}
@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<GridFieldPB> _fields;
const GridFieldEquatable(
UnmodifiableListView<GridFieldPB> fields,
) : _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<GridFieldPB> get value => UnmodifiableListView(_fields);
}
2022-08-10 12:58:07 +08:00
class TextItem extends ColumnItem {
final String s;
TextItem(this.s);
@override
String get id => s;
}
class RichTextItem extends ColumnItem {
final String title;
final String subtitle;
RichTextItem({required this.title, required this.subtitle});
@override
String get id => title;
}