nc-photos/app/lib/widget/home_photos/bloc.dart
2024-02-07 01:14:30 +08:00

334 lines
9.7 KiB
Dart

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,
}) : 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<_StartScaling>(_onStartScaling);
on<_EndScaling>(_onEndScaling);
on<_SetScale>(_onSetScale);
on<_SetEnableMemoryCollection>(_onSetEnableMemoryCollection);
on<_SetError>(_onSetError);
_subscriptions
.add(accountPrefController.isEnableMemoryAlbum.listen((event) {
add(_SetEnableMemoryCollection(event));
}));
}
@override
Future<void> close() {
for (final s in _subscriptions) {
s.cancel();
}
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;
};
@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(_LoadItems ev, Emitter<_State> emit) {
_log.info(ev);
return emit.forEach<FilesStreamEvent>(
controller.stream,
onData: (data) => 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);
unawaited(controller.reload());
}
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,
));
}
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 _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;
if (state.scale! >= 1.25) {
// scale up
newZoom = (state.zoom + 1).clamp(-1, 2);
} else if (state.scale! <= 0.75) {
newZoom = (state.zoom - 1).clamp(-1, 2);
} else {
newZoom = state.zoom;
}
emit(state.copyWith(
zoom: newZoom,
scale: null,
));
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 _onSetError(_SetError ev, Emitter<_State> emit) {
_log.info(ev);
emit(state.copyWith(error: ExceptionEvent(ev.error, ev.stackTrace)));
}
Future _transformItems(List<FileDescriptor> files) async {
_log.info("[_transformItems] Queue ${files.length} items");
_itemTransformerQueue.addJob(
_ItemTransformerArgument(
account: account,
files: files,
memoriesDayRange: prefController.memoriesRange.value,
locale: language_util.getSelectedLocale() ??
PlatformDispatcher.instance.locale,
),
_buildItem,
(result) {
if (!isClosed) {
add(_OnItemTransformed(result.items, result.memoryCollections));
}
},
);
}
void _clearSelection(Emitter<_State> emit) {
emit(state.copyWith(selectedItems: const {}));
}
final DiContainer _c;
final Account account;
final FilesController controller;
final PrefController prefController;
final AccountPrefController accountPrefController;
final CollectionsController collectionsController;
final _itemTransformerQueue =
ComputeQueue<_ItemTransformerArgument, _ItemTransformerResult>();
final _subscriptions = <StreamSubscription>[];
var _isHandlingError = false;
}
_ItemTransformerResult _buildItem(_ItemTransformerArgument arg) {
final sortedFiles = arg.files
.where((f) => f.fdIsArchived != true)
.sorted(compareFileDescriptorDateTimeDescending);
final dateHelper = photo_list_util.DateGroupHelper(isMonthOnly: false);
final today = clock.now();
final memoryCollectionHelper = photo_list_util.MemoryCollectionHelper(
arg.account,
today: today,
dayRange: arg.memoriesDayRange,
);
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));
}
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;
}
}