nc-photos/app/lib/widget/home_collections/bloc.dart

130 lines
3.8 KiB
Dart

part of '../home_collections.dart';
@npLog
class _Bloc extends Bloc<_Event, _State> implements BlocLogger {
_Bloc({
required this.account,
required this.controller,
required this.prefController,
}) : super(_State.init()) {
on<_LoadCollections>(_onLoad);
on<_ReloadCollections>(_onReload);
on<_TransformItems>(_onTransformItems);
on<_SetSelectedItems>(_onSetSelectedItems);
on<_RemoveSelectedItems>(_onRemoveSelectedItems);
on<_UpdateCollectionSort>(_onUpdateCollectionSort);
on<_SetCollectionSort>(_onSetCollectionSort);
on<_SetError>(_onSetError);
_homeAlbumsSortSubscription =
prefController.homeAlbumsSort.distinct().listen((event) {
add(_UpdateCollectionSort(collection_util.CollectionSort.values[event]));
});
}
@override
Future<void> close() {
_homeAlbumsSortSubscription?.cancel();
return super.close();
}
@override
String get tag => _log.fullName;
@override
bool Function(dynamic, dynamic)? get shouldLog => null;
@override
void onError(Object error, StackTrace stackTrace) {
// we need this to prevent onError being triggered recursively
if (!isClosed && !_isHandlingError) {
_isHandlingError = true;
try {
add(_SetError(error, stackTrace));
} catch (_) {}
_isHandlingError = false;
}
super.onError(error, stackTrace);
}
Future<void> _onLoad(_LoadCollections ev, Emitter<_State> emit) async {
_log.info("[_onLoad] $ev");
return emit.forEach<CollectionStreamEvent>(
controller.stream,
onData: (data) => state.copyWith(
collections: data.data.map((d) => d.collection).toList(),
isLoading: data.hasNext,
),
onError: (e, stackTrace) {
_log.severe("[_onLoad] Uncaught exception", e, stackTrace);
return state.copyWith(
isLoading: false,
error: ExceptionEvent(e, stackTrace),
);
},
);
}
void _onReload(_ReloadCollections ev, Emitter<_State> emit) {
_log.info("[_onReload] $ev");
unawaited(controller.reload());
}
Future<void> _onTransformItems(
_TransformItems ev, Emitter<_State> emit) async {
_log.info("[_onTransformItems] $ev");
final transformed = _transformCollections(ev.collections, state.sort);
emit(state.copyWith(transformedItems: transformed));
}
void _onSetSelectedItems(_SetSelectedItems ev, Emitter<_State> emit) {
_log.info("[_onSetSelectedItems] $ev");
emit(state.copyWith(selectedItems: ev.items));
}
void _onRemoveSelectedItems(_RemoveSelectedItems ev, Emitter<_State> emit) {
_log.info("[_onDeleteSelectedItems] $ev");
final selected = state.selectedItems;
emit(state.copyWith(selectedItems: const {}));
controller.remove(selected.map((e) => e.collection).toList());
}
void _onUpdateCollectionSort(_UpdateCollectionSort ev, Emitter<_State> emit) {
_log.info("[_onUpdateCollectionSort] $ev");
if (ev.sort != state.sort) {
final transformed = _transformCollections(state.collections, ev.sort);
emit(state.copyWith(
transformedItems: transformed,
sort: ev.sort,
));
}
}
void _onSetCollectionSort(_SetCollectionSort ev, Emitter<_State> emit) {
_log.info("[_onSetCollectionSort] $ev");
prefController.setHomeAlbumsSort(ev.sort.index);
}
void _onSetError(_SetError ev, Emitter<_State> emit) {
_log.info(ev);
emit(state.copyWith(error: ExceptionEvent(ev.error, ev.stackTrace)));
}
List<_Item> _transformCollections(
List<Collection> collections,
collection_util.CollectionSort sort,
) {
final sorted = collections.sortedBy(sort);
return sorted.map((c) => _Item(c)).toList();
}
final Account account;
final CollectionsController controller;
final PrefController prefController;
StreamSubscription<int>? _homeAlbumsSortSubscription;
var _isHandlingError = false;
}