part of '../home_photos2.dart'; @npLog class _Bloc extends Bloc<_Event, _State> with BlocLogger { _Bloc( this._c, { required this.account, required this.controller, required this.prefController, required this.accountPrefController, required this.collectionsController, required this.sessionController, }) : super(_State.init( zoom: prefController.homePhotosZoomLevel.value, isEnableMemoryCollection: accountPrefController.isEnableMemoryAlbum.value, )) { on<_LoadItems>(_onLoad); on<_Reload>(_onReload); on<_TransformItems>(_onTransformItems); on<_OnItemTransformed>(_onOnItemTransformed); on<_SetSelectedItems>(_onSetSelectedItems); on<_AddSelectedItemsToCollection>(_onAddSelectedItemsToCollection); on<_ArchiveSelectedItems>(_onArchiveSelectedItems); on<_DeleteSelectedItems>(_onDeleteSelectedItems); on<_DownloadSelectedItems>(_onDownloadSelectedItems); on<_AddVisibleItem>(_onAddVisibleItem); on<_RemoveVisibleItem>(_onRemoveVisibleItem); on<_SetContentListMaxExtent>(_onSetContentListMaxExtent); on<_SetSyncProgress>(_onSetSyncProgress); on<_StartScaling>(_onStartScaling); on<_EndScaling>(_onEndScaling); on<_SetScale>(_onSetScale); on<_SetEnableMemoryCollection>(_onSetEnableMemoryCollection); on<_SetSortByName>(_onSetSortByName); on<_SetMemoriesRange>(_onSetMemoriesRange); on<_SetEnableExif>(_onSetEnableExif); on<_UpdateDateTimeGroup>(_onUpdateDateTimeGroup); on<_SetError>(_onSetError); _subscriptions .add(accountPrefController.isEnableMemoryAlbum.listen((event) { add(_SetEnableMemoryCollection(event)); })); _subscriptions.add(prefController.isPhotosTabSortByName.listen((event) { add(_SetSortByName(event)); })); _subscriptions.add(prefController.memoriesRange.listen((event) { add(_SetMemoriesRange(event)); })); _subscriptions.add(prefController.isEnableExif.listen((event) { add(_SetEnableExif(event)); })); _nativeFileExifUpdatedListener?.begin(); } @override Future close() { for (final s in _subscriptions) { s.cancel(); } _nativeFileExifUpdatedListener?.end(); return super.close(); } @override String get tag => _log.fullName; @override bool Function(dynamic, dynamic)? get shouldLog => (currentState, nextState) { currentState = currentState as _State; nextState = nextState as _State; return currentState.scale == nextState.scale && currentState.visibleItems == nextState.visibleItems && currentState.syncProgress == nextState.syncProgress; }; @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(_LoadItems ev, Emitter<_State> emit) { _log.info(ev); return emit.forEach( controller.stream, onData: (data) { if (_isInitialLoad && !data.hasNext) { _isInitialLoad = false; _syncRemote(); } return state.copyWith( files: data.data, isLoading: data.hasNext || _itemTransformerQueue.isProcessing, ); }, onError: (e, stackTrace) { _log.severe("[_onLoad] Uncaught exception", e, stackTrace); return state.copyWith( isLoading: _itemTransformerQueue.isProcessing, error: ExceptionEvent(e, stackTrace), ); }, ); } void _onReload(_Reload ev, Emitter<_State> emit) { _log.info(ev); emit(state.copyWith(syncProgress: const Progress(0))); _syncRemote(); } void _onTransformItems(_TransformItems ev, Emitter<_State> emit) { _log.info(ev); _transformItems(ev.items); emit(state.copyWith(isLoading: true)); } void _onOnItemTransformed(_OnItemTransformed ev, Emitter<_State> emit) { _log.info(ev); emit(state.copyWith( transformedItems: ev.items, memoryCollections: ev.memoryCollections, isLoading: _itemTransformerQueue.isProcessing, )); _tryStartMetadataTask(); } void _onSetSelectedItems(_SetSelectedItems ev, Emitter<_State> emit) { _log.info(ev); emit(state.copyWith(selectedItems: ev.items)); } void _onAddSelectedItemsToCollection( _AddSelectedItemsToCollection ev, Emitter<_State> emit) { _log.info(ev); final selected = state.selectedItems; _clearSelection(emit); final selectedFiles = selected.whereType<_FileItem>().map((e) => e.file).toList(); if (selectedFiles.isNotEmpty) { final targetController = collectionsController.stream.value .itemsControllerByCollection(ev.collection); targetController.addFiles(selectedFiles).onError((e, stackTrace) { if (e != null) { add(_SetError(e, stackTrace)); } }); } } void _onArchiveSelectedItems(_ArchiveSelectedItems ev, Emitter<_State> emit) { _log.info(ev); final selected = state.selectedItems; _clearSelection(emit); final selectedFiles = selected.whereType<_FileItem>().map((e) => e.file).toList(); if (selectedFiles.isNotEmpty) { controller.updateProperty( selectedFiles, isArchived: const OrNull(true), errorBuilder: (fileIds) => _ArchiveFailedError(fileIds.length), ); } } void _onDeleteSelectedItems(_DeleteSelectedItems ev, Emitter<_State> emit) { _log.info(ev); final selected = state.selectedItems; _clearSelection(emit); final selectedFiles = selected.whereType<_FileItem>().map((e) => e.file).toList(); if (selectedFiles.isNotEmpty) { controller.remove( selectedFiles, errorBuilder: (fileIds) => _RemoveFailedError(fileIds.length), ); } } void _onDownloadSelectedItems( _DownloadSelectedItems ev, Emitter<_State> emit) { _log.info(ev); final selected = state.selectedItems; _clearSelection(emit); final selectedFiles = selected.whereType<_FileItem>().map((e) => e.file).toList(); if (selectedFiles.isNotEmpty) { unawaited(DownloadHandler(_c).downloadFiles(account, selectedFiles)); } } void _onAddVisibleItem(_AddVisibleItem ev, Emitter<_State> emit) { // _log.info(ev); if (state.visibleItems.contains(ev.item)) { return; } emit(state.copyWith( visibleItems: state.visibleItems.added(ev.item), )); } void _onRemoveVisibleItem(_RemoveVisibleItem ev, Emitter<_State> emit) { // _log.info(ev); if (!state.visibleItems.contains(ev.item)) { return; } emit(state.copyWith( visibleItems: state.visibleItems.removed(ev.item), )); } void _onSetContentListMaxExtent( _SetContentListMaxExtent ev, Emitter<_State> emit) { _log.info(ev); emit(state.copyWith(contentListMaxExtent: ev.value)); } void _onSetSyncProgress(_SetSyncProgress ev, Emitter<_State> emit) { _log.info(ev); emit(state.copyWith(syncProgress: ev.progress)); } void _onStartScaling(_StartScaling ev, Emitter<_State> emit) { _log.info(ev); } void _onEndScaling(_EndScaling ev, Emitter<_State> emit) { _log.info(ev); if (state.scale == null) { return; } final int newZoom; final currZoom = state.zoom; if (state.scale! >= 1.25) { // scale up newZoom = (currZoom + 1).clamp(-1, 2); } else if (state.scale! <= 0.75) { newZoom = (currZoom - 1).clamp(-1, 2); } else { newZoom = currZoom; } emit(state.copyWith( zoom: newZoom, scale: null, )); if ((currZoom >= 0) != (newZoom >= 0)) { add(const _UpdateDateTimeGroup()); } unawaited(prefController.setHomePhotosZoomLevel(newZoom)); } void _onSetScale(_SetScale ev, Emitter<_State> emit) { // _log.info(ev); emit(state.copyWith(scale: ev.scale)); } void _onSetEnableMemoryCollection( _SetEnableMemoryCollection ev, Emitter<_State> emit) { _log.info(ev); emit(state.copyWith(isEnableMemoryCollection: ev.value)); } void _onSetSortByName(_SetSortByName ev, Emitter<_State> emit) { _log.info(ev); _transformItems(state.files); } void _onSetMemoriesRange(_SetMemoriesRange ev, Emitter<_State> emit) { _log.info(ev); _transformItems(state.files); } void _onSetEnableExif(_SetEnableExif ev, Emitter<_State> emit) { _log.info(ev); if (ev.value) { _tryStartMetadataTask(ignoreFired: true); } else { _stopMetadataTask(); } } void _onUpdateDateTimeGroup(_UpdateDateTimeGroup ev, Emitter<_State> emit) { _log.info(ev); _transformItems(state.files); } void _onSetError(_SetError ev, Emitter<_State> emit) { _log.info(ev); emit(state.copyWith(error: ExceptionEvent(ev.error, ev.stackTrace))); } void _onNativeFileExifUpdated(FileExifUpdatedEvent ev) { _log.info(ev); _refreshThrottler.trigger( maxResponceTime: const Duration(seconds: 3), maxPendingCount: 10, ); } Future _transformItems(List files) async { _log.info("[_transformItems] Queue ${files.length} items"); _itemTransformerQueue.addJob( _ItemTransformerArgument( account: account, files: files, sort: prefController.isPhotosTabSortByName.value ? _ItemSort.filename : _ItemSort.dateTime, isGroupByDay: prefController.homePhotosZoomLevel.value >= 0, memoriesDayRange: prefController.memoriesRange.value, locale: language_util.getSelectedLocale() ?? PlatformDispatcher.instance.locale, ), _buildItem, (result) { if (!isClosed) { add(_OnItemTransformed(result.items, result.memoryCollections)); } }, ); } void _syncRemote() { final stopwatch = Stopwatch()..start(); controller.syncRemote( onProgressUpdate: (progress) { if (!isClosed) { add(_SetSyncProgress(progress)); } }, ).whenComplete(() { if (!isClosed) { add(const _SetSyncProgress(null)); } _log.info( "[_syncRemote] Elapsed time: ${stopwatch.elapsedMilliseconds}ms"); }); } void _clearSelection(Emitter<_State> emit) { emit(state.copyWith(selectedItems: const {})); } Future _tryStartMetadataTask({ bool ignoreFired = false, }) async { if (state.files.isNotEmpty && prefController.isEnableExif.value && (ignoreFired || !sessionController.hasFiredMetadataTask.value)) { sessionController.setFiredMetadataTask(true); try { final missingMetadataCount = await _c.npDb.countFilesByFileIdsMissingMetadata( account: account.toDb(), fileIds: state.files.map((e) => e.fdId).toList(), mimes: file_util.supportedImageFormatMimes, ); _log.info( "[_tryStartMetadataTask] Missing count: $missingMetadataCount"); if (missingMetadataCount > 0) { unawaited(service.startService()); } } catch (e, stackTrace) { _log.shout("[_tryStartMetadataTask] Failed starting metadata task", e, stackTrace); } } } void _stopMetadataTask() { service.stopService(); } final DiContainer _c; final Account account; final FilesController controller; final PrefController prefController; final AccountPrefController accountPrefController; final CollectionsController collectionsController; final SessionController sessionController; final _itemTransformerQueue = ComputeQueue<_ItemTransformerArgument, _ItemTransformerResult>(); final _subscriptions = []; var _isHandlingError = false; var _isInitialLoad = true; // Listen to updates from background isolates as the memories are not shared late final _nativeFileExifUpdatedListener = getRawPlatform() == NpPlatform.android ? NativeEventListener(_onNativeFileExifUpdated) : null; late final _refreshThrottler = Throttler( onTriggered: (_) { add(const _Reload()); }, logTag: _log.name, ); } _ItemTransformerResult _buildItem(_ItemTransformerArgument arg) { final int Function(FileDescriptor, FileDescriptor) sorter; switch (arg.sort) { case _ItemSort.filename: sorter = (a, b) => a.fdPath.compareTo(b.fdPath); break; case _ItemSort.dateTime: default: sorter = compareFileDescriptorDateTimeDescending; break; } final sortedFiles = arg.files.where((f) => f.fdIsArchived != true).sorted(sorter); final dateHelper = arg.sort == _ItemSort.dateTime ? photo_list_util.DateGroupHelper(isMonthOnly: !arg.isGroupByDay) : null; final today = clock.now(); final memoryCollectionHelper = arg.sort == _ItemSort.dateTime ? photo_list_util.MemoryCollectionHelper( arg.account, today: today, dayRange: arg.memoriesDayRange, ) : null; final transformed = <_Item>[]; for (int i = 0; i < sortedFiles.length; ++i) { final file = sortedFiles[i]; final item = _buildSingleItem(arg.account, file); if (item == null) { continue; } final date = dateHelper?.onFile(file); if (date != null) { transformed.add(_DateItem(date: date, isMonthOnly: !arg.isGroupByDay)); } transformed.add(item); memoryCollectionHelper?.addFile(file); } final memoryCollections = memoryCollectionHelper ?.build((year) => L10n.of(arg.locale).memoryAlbumName(today.year - year)); return _ItemTransformerResult( items: transformed, memoryCollections: memoryCollections ?? [], ); } _Item? _buildSingleItem(Account account, FileDescriptor file) { if (file_util.isSupportedImageFormat(file)) { return _PhotoItem( file: file, account: account, ); } else if (file_util.isSupportedVideoFormat(file)) { return _VideoItem( file: file, account: account, ); } else { _$__NpLog.log .shout("[_buildSingleItem] Unsupported file format: ${file.fdMime}"); return null; } }