188 lines
5.4 KiB
Dart
Raw Normal View History

2022-03-03 10:51:52 +08:00
import 'dart:async';
import 'package:dartz/dartz.dart';
import 'package:equatable/equatable.dart';
2022-06-24 15:23:39 +08:00
import 'package:flowy_sdk/log.dart';
2022-03-03 10:51:52 +08:00
import 'package:flowy_sdk/protobuf/flowy-error/errors.pb.dart';
2022-07-04 15:00:54 +08:00
import 'package:flowy_sdk/protobuf/flowy-folder/view.pb.dart';
import 'package:flowy_sdk/protobuf/flowy-grid/protobuf.dart';
2022-03-03 10:51:52 +08:00
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:freezed_annotation/freezed_annotation.dart';
import 'block/block_cache.dart';
2022-03-03 10:51:52 +08:00
import 'grid_service.dart';
import 'row/row_service.dart';
import 'dart:collection';
2022-03-03 10:51:52 +08:00
part 'grid_bloc.freezed.dart';
class GridBloc extends Bloc<GridEvent, GridState> {
2022-07-03 15:53:28 +08:00
final String gridId;
2022-04-10 08:25:01 +08:00
final GridService _gridService;
2022-04-14 13:29:42 +08:00
final GridFieldCache fieldCache;
2022-03-03 10:51:52 +08:00
2022-07-03 15:53:28 +08:00
// key: the block id
2022-07-16 11:53:39 +08:00
final LinkedHashMap<String, GridBlockCache> _blocks;
2022-07-03 15:53:28 +08:00
List<GridRowInfo> get rowInfos {
final List<GridRowInfo> rows = [];
2022-07-03 15:53:28 +08:00
for (var block in _blocks.values) {
rows.addAll(block.rows);
}
return rows;
}
2022-06-24 15:23:39 +08:00
2022-07-19 14:11:29 +08:00
GridBloc({required ViewPB view})
2022-07-03 15:53:28 +08:00
: gridId = view.id,
_blocks = LinkedHashMap.identity(),
_gridService = GridService(gridId: view.id),
fieldCache = GridFieldCache(gridId: view.id),
2022-04-10 16:29:45 +08:00
super(GridState.initial(view.id)) {
2022-03-03 10:51:52 +08:00
on<GridEvent>(
(event, emit) async {
await event.when(
initial: () async {
2022-04-10 08:25:01 +08:00
_startListening();
2022-04-14 13:29:42 +08:00
await _loadGrid(emit);
2022-03-03 10:51:52 +08:00
},
createRow: () {
2022-04-10 16:29:45 +08:00
_gridService.createRow();
2022-03-03 10:51:52 +08:00
},
didReceiveRowUpdate: (newRowInfos, reason) {
emit(state.copyWith(rowInfos: newRowInfos, reason: reason));
2022-03-18 17:14:46 +08:00
},
didReceiveFieldUpdate: (fields) {
emit(state.copyWith(rowInfos: rowInfos, fields: GridFieldEquatable(fields)));
},
2022-03-03 10:51:52 +08:00
);
},
);
}
@override
Future<void> close() async {
2022-04-10 16:29:45 +08:00
await _gridService.closeGrid();
2022-04-25 22:03:10 +08:00
await fieldCache.dispose();
2022-07-03 15:53:28 +08:00
for (final blockCache in _blocks.values) {
blockCache.dispose();
}
2022-03-03 10:51:52 +08:00
return super.close();
}
GridRowCache? getRowCache(String blockId, String rowId) {
2022-07-16 11:53:39 +08:00
final GridBlockCache? blockCache = _blocks[blockId];
2022-07-03 15:53:28 +08:00
return blockCache?.rowCache;
}
2022-04-14 13:29:42 +08:00
void _startListening() {
fieldCache.addListener(
listenWhen: () => !isClosed,
2022-07-03 15:53:28 +08:00
onFields: (fields) => add(GridEvent.didReceiveFieldUpdate(fields)),
);
2022-03-26 20:27:32 +08:00
}
2022-03-03 22:17:07 +08:00
Future<void> _loadGrid(Emitter<GridState> emit) async {
2022-04-10 16:29:45 +08:00
final result = await _gridService.loadGrid();
2022-03-18 17:14:46 +08:00
return Future(
() => result.fold(
2022-06-24 15:23:39 +08:00
(grid) async {
2022-07-03 15:53:28 +08:00
_initialBlocks(grid.blocks);
2022-06-24 15:23:39 +08:00
await _loadFields(grid, emit);
},
2022-03-18 17:14:46 +08:00
(err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
),
2022-03-03 10:51:52 +08:00
);
}
2022-07-17 14:13:12 +08:00
Future<void> _loadFields(GridPB grid, Emitter<GridState> emit) async {
final result = await _gridService.getFields(fieldIds: grid.fields);
2022-03-18 17:14:46 +08:00
return Future(
() => result.fold(
2022-03-20 17:17:06 +08:00
(fields) {
fieldCache.fields = fields.items;
2022-04-14 13:29:42 +08:00
2022-03-20 17:17:06 +08:00
emit(state.copyWith(
grid: Some(grid),
fields: GridFieldEquatable(fieldCache.fields),
rowInfos: rowInfos,
2022-03-20 17:17:06 +08:00
loadingState: GridLoadingState.finish(left(unit)),
));
},
2022-03-18 17:14:46 +08:00
(err) => emit(state.copyWith(loadingState: GridLoadingState.finish(right(err)))),
),
2022-03-17 20:59:21 +08:00
);
2022-03-03 22:17:07 +08:00
}
2022-07-03 15:53:28 +08:00
2022-07-17 14:13:12 +08:00
void _initialBlocks(List<GridBlockPB> blocks) {
2022-07-03 15:53:28 +08:00
for (final block in blocks) {
if (_blocks[block.id] != null) {
Log.warn("Intial duplicate block's cache: ${block.id}");
return;
}
2022-07-16 11:53:39 +08:00
final cache = GridBlockCache(
2022-07-03 15:53:28 +08:00
gridId: gridId,
block: block,
fieldCache: fieldCache,
);
cache.addListener(
listenWhen: () => !isClosed,
onChangeReason: (reason) => add(GridEvent.didReceiveRowUpdate(rowInfos, reason)),
2022-07-03 15:53:28 +08:00
);
_blocks[block.id] = cache;
}
}
2022-03-03 10:51:52 +08:00
}
@freezed
2022-03-25 15:02:43 +08:00
class GridEvent with _$GridEvent {
2022-03-08 15:25:56 +08:00
const factory GridEvent.initial() = InitialGrid;
2022-03-03 10:51:52 +08:00
const factory GridEvent.createRow() = _CreateRow;
const factory GridEvent.didReceiveRowUpdate(List<GridRowInfo> rows, GridRowChangeReason listState) =
_DidReceiveRowUpdate;
2022-07-17 14:13:12 +08:00
const factory GridEvent.didReceiveFieldUpdate(List<GridFieldPB> fields) = _DidReceiveFieldUpdate;
2022-03-03 10:51:52 +08:00
}
@freezed
2022-03-25 15:02:43 +08:00
class GridState with _$GridState {
2022-03-03 10:51:52 +08:00
const factory GridState({
2022-04-10 16:29:45 +08:00
required String gridId,
2022-07-17 14:13:12 +08:00
required Option<GridPB> grid,
required GridFieldEquatable fields,
required List<GridRowInfo> rowInfos,
required GridLoadingState loadingState,
2022-07-03 15:53:28 +08:00
required GridRowChangeReason reason,
2022-03-03 10:51:52 +08:00
}) = _GridState;
2022-04-10 16:29:45 +08:00
factory GridState.initial(String gridId) => GridState(
fields: const GridFieldEquatable([]),
rowInfos: [],
2022-03-17 20:59:21 +08:00
grid: none(),
2022-04-10 16:29:45 +08:00
gridId: gridId,
loadingState: const _Loading(),
2022-07-03 15:53:28 +08:00
reason: const InitialListState(),
2022-03-03 10:51:52 +08:00
);
}
@freezed
class GridLoadingState with _$GridLoadingState {
const factory GridLoadingState.loading() = _Loading;
const factory GridLoadingState.finish(Either<Unit, FlowyError> successOrFail) = _Finish;
}
class GridFieldEquatable extends Equatable {
2022-07-17 14:13:12 +08:00
final List<GridFieldPB> _fields;
const GridFieldEquatable(List<GridFieldPB> fields) : _fields = fields;
@override
List<Object?> get props {
return [
_fields.length,
_fields.map((field) => field.width).reduce((value, element) => value + element),
];
}
2022-07-17 14:13:12 +08:00
UnmodifiableListView<GridFieldPB> get value => UnmodifiableListView(_fields);
}