part of '../home_collections.dart'; @npLog class _Bloc extends Bloc<_Event, _State> implements BlocTag { _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 close() { _homeAlbumsSortSubscription?.cancel(); return super.close(); } @override String get tag => _log.fullName; @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 _onLoad(_LoadCollections ev, Emitter<_State> emit) async { _log.info("[_onLoad] $ev"); return emit.forEach( 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 _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 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? _homeAlbumsSortSubscription; var _isHandlingError = false; }