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

537 lines
16 KiB
Dart
Raw Normal View History

part of '../collection_browser.dart';
@npLog
2023-07-28 18:48:50 +02:00
class _Bloc extends Bloc<_Event, _State> with BlocLogger {
_Bloc({
required DiContainer container,
required this.account,
required this.prefController,
required this.collectionsController,
required Collection collection,
}) : _c = container,
_isAdHocCollection = !collectionsController.stream.value.data
.any((e) => e.collection.compareIdentity(collection)),
super(_State.init(
collection: collection,
coverUrl: _getCoverUrl(collection),
zoom: prefController.albumBrowserZoomLevel.value,
)) {
_initItemController(collection);
on<_UpdateCollection>(_onUpdateCollection);
on<_LoadItems>(_onLoad);
on<_TransformItems>(_onTransformItems);
on<_ImportPendingSharedCollection>(_onImportPendingSharedCollection);
2023-04-15 07:26:19 +02:00
on<_Download>(_onDownload);
on<_BeginEdit>(_onBeginEdit);
on<_EditName>(_onEditName);
on<_AddLabelToCollection>(_onAddLabelToCollection);
on<_EditSort>(_onEditSort);
on<_EditManualSort>(_onEditManualSort);
on<_TransformEditItems>(_onTransformEditItems);
on<_DoneEdit>(_onDoneEdit, transformer: concurrent());
on<_CancelEdit>(_onCancelEdit);
2023-04-17 18:15:29 +02:00
on<_UnsetCover>(_onUnsetCover);
on<_SetSelectedItems>(_onSetSelectedItems);
on<_DownloadSelectedItems>(_onDownloadSelectedItems);
on<_AddSelectedItemsToCollection>(_onAddSelectedItemsToCollection);
on<_RemoveSelectedItemsFromCollection>(
_onRemoveSelectedItemsFromCollection);
on<_ArchiveSelectedItems>(_onArchiveSelectedItems,
transformer: concurrent());
on<_DeleteSelectedItems>(_onDeleteSelectedItems);
on<_SetDragging>(_onSetDragging);
on<_StartScaling>(_onStartScaling);
on<_EndScaling>(_onEndScaling);
on<_SetScale>(_onSetScale);
on<_SetError>(_onSetError);
on<_SetMessage>(_onSetMessage);
if (!_isAdHocCollection) {
_collectionControllerSubscription =
collectionsController.stream.listen((event) {
final c = event.data
.firstWhere((d) => state.collection.compareIdentity(d.collection));
if (!identical(c, state.collection)) {
add(_UpdateCollection(c.collection));
}
});
} else {
_log.info("[_Bloc] Ad hoc collection");
}
_itemsControllerSubscription = itemsController.stream.listen(
(_) {},
onError: (e, stackTrace) {
add(_SetError(e, stackTrace));
},
);
}
@override
Future<void> close() {
_collectionControllerSubscription?.cancel();
_itemsControllerSubscription?.cancel();
return super.close();
}
@override
bool Function(dynamic, dynamic)? get shouldLog => (currentState, nextState) {
return currentState.scale == nextState.scale;
};
2023-05-20 13:07:07 +02:00
@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);
}
2023-04-17 18:15:29 +02:00
bool isCollectionCapabilityPermitted(CollectionCapability capability) {
return CollectionAdapter.of(_c, account, state.collection)
.isPermitted(capability);
}
void _onUpdateCollection(_UpdateCollection ev, Emitter<_State> emit) {
_log.info("$ev");
emit(state.copyWith(collection: ev.collection));
2023-04-17 18:15:29 +02:00
_updateCover(emit);
}
Future<void> _onLoad(_LoadItems ev, Emitter<_State> emit) {
_log.info("$ev");
return emit.forEach<CollectionItemStreamData>(
itemsController.stream.handleError((e, stackTrace) {
_log.severe("[_onLoad] Uncaught exception", e, stackTrace);
return state.copyWith(
isLoading: false,
error: ExceptionEvent(e, stackTrace),
);
}),
onData: (data) => state.copyWith(
items: data.items,
isLoading: data.hasNext,
),
);
}
void _onTransformItems(_TransformItems ev, Emitter<_State> emit) {
_log.info("$ev");
final result = _transformItems(ev.items, state.collection.itemSort);
2023-04-17 18:15:29 +02:00
emit(state.copyWith(transformedItems: result.transformed));
_updateCover(emit);
}
2023-04-15 07:26:19 +02:00
void _onDownload(_Download ev, Emitter<_State> emit) {
_log.info("$ev");
if (state.items.isNotEmpty) {
unawaited(DownloadHandler(_c).downloadFiles(
account,
state.items.whereType<CollectionFileItem>().map((e) => e.file).toList(),
parentDir: state.collection.name,
));
}
}
Future<void> _onImportPendingSharedCollection(
_ImportPendingSharedCollection ev, Emitter<_State> emit) async {
_log.info(ev);
final newCollection = await collectionsController
.importPendingSharedCollection(state.collection);
if (newCollection != null) {
emit(state.copyWith(importResult: newCollection));
}
}
void _onBeginEdit(_BeginEdit ev, Emitter<_State> emit) {
_log.info("$ev");
emit(state.copyWith(isEditMode: true));
}
void _onEditName(_EditName ev, Emitter<_State> emit) {
_log.info("$ev");
emit(state.copyWith(editName: ev.name));
}
void _onAddLabelToCollection(_AddLabelToCollection ev, Emitter<_State> emit) {
_log.info("$ev");
2023-04-17 18:15:29 +02:00
assert(isCollectionCapabilityPermitted(CollectionCapability.labelItem));
emit(state.copyWith(
editItems: [
NewCollectionLabelItem(ev.label, clock.now().toUtc()),
...state.editItems ?? state.items,
],
));
}
void _onEditSort(_EditSort ev, Emitter<_State> emit) {
_log.info("$ev");
final result = _transformItems(state.editItems ?? state.items, ev.sort);
emit(state.copyWith(
editSort: ev.sort,
editTransformedItems: result.transformed,
));
}
void _onEditManualSort(_EditManualSort ev, Emitter<_State> emit) {
_log.info("$ev");
2023-04-17 18:15:29 +02:00
assert(isCollectionCapabilityPermitted(CollectionCapability.manualSort));
emit(state.copyWith(
editSort: CollectionItemSort.manual,
editItems:
ev.sorted.whereType<_ActualItem>().map((e) => e.original).toList(),
editTransformedItems: ev.sorted,
));
}
void _onTransformEditItems(_TransformEditItems ev, Emitter<_State> emit) {
_log.info("$ev");
final result =
_transformItems(ev.items, state.editSort ?? state.collection.itemSort);
emit(state.copyWith(editTransformedItems: result.transformed));
}
Future<void> _onDoneEdit(_DoneEdit ev, Emitter<_State> emit) async {
_log.info("$ev");
emit(state.copyWith(isEditBusy: true));
try {
await collectionsController.edit(
state.collection,
name: state.editName,
items: state.editItems,
itemSort: state.editSort,
);
emit(state.copyWith(
isEditMode: false,
isEditBusy: false,
editName: null,
editItems: null,
editTransformedItems: null,
editSort: null,
));
} catch (e, stackTrace) {
_log.severe("[_onDoneEdit] Failed while edit", e, stackTrace);
emit(state.copyWith(
error: ExceptionEvent(e, stackTrace),
isEditBusy: false,
));
}
}
void _onCancelEdit(_CancelEdit ev, Emitter<_State> emit) {
_log.info("$ev");
emit(state.copyWith(
isEditMode: false,
editName: null,
editItems: null,
editTransformedItems: null,
editSort: null,
));
}
2023-04-17 18:15:29 +02:00
void _onUnsetCover(_UnsetCover ev, Emitter<_State> emit) {
_log.info("$ev");
collectionsController.edit(state.collection, cover: const OrNull(null));
2023-04-17 18:15:29 +02:00
}
void _onSetSelectedItems(_SetSelectedItems ev, Emitter<_State> emit) {
_log.info("$ev");
2023-04-17 18:15:29 +02:00
final adapter = CollectionAdapter.of(_c, account, state.collection);
emit(state.copyWith(
selectedItems: ev.items,
2023-04-17 18:15:29 +02:00
isSelectionRemovable: ev.items
.whereType<_ActualItem>()
.map((e) => e.original)
.any(adapter.isItemRemovable),
));
}
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 _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 _onRemoveSelectedItemsFromCollection(
_RemoveSelectedItemsFromCollection ev, Emitter<_State> emit) {
_log.info("$ev");
final selected = state.selectedItems;
_clearSelection(emit);
final selectedItems =
selected.whereType<_ActualItem>().map((e) => e.original).toList();
if (selectedItems.isNotEmpty) {
unawaited(itemsController.removeItems(selectedItems));
}
}
Future<void> _onArchiveSelectedItems(
_ArchiveSelectedItems ev, Emitter<_State> emit) async {
_log.info("$ev");
final selected = state.selectedItems;
_clearSelection(emit);
final selectedFds =
selected.whereType<_FileItem>().map((e) => e.file).toList();
if (selectedFds.isNotEmpty) {
final selectedFiles =
await InflateFileDescriptor(_c)(account, selectedFds);
final count = await ArchiveFile(_c)(account, selectedFiles);
if (count != selectedFiles.length) {
emit(state.copyWith(
message: L10n.global()
.archiveSelectedFailureNotification(selectedFiles.length - count),
));
}
}
}
Future<void> _onDeleteSelectedItems(
_DeleteSelectedItems ev, Emitter<_State> emit) async {
_log.info("$ev");
final selected = state.selectedItems;
_clearSelection(emit);
final selectedFiles =
selected.whereType<_FileItem>().map((e) => e.file).toList();
if (selectedFiles.isNotEmpty) {
final count = await Remove(_c)(
account,
selectedFiles,
onError: (_, f, e, stackTrace) {
_log.severe(
"[_onDeleteSelectedItems] Failed while Remove: ${logFilename(f.strippedPath)}",
e,
stackTrace,
);
},
);
if (count != selectedFiles.length) {
emit(state.copyWith(
message: L10n.global()
.deleteSelectedFailureNotification(selectedFiles.length - count),
));
}
}
}
void _onSetDragging(_SetDragging ev, Emitter<_State> emit) {
_log.info("$ev");
emit(state.copyWith(isDragging: ev.flag));
}
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.setAlbumBrowserZoomLevel(newZoom));
}
void _onSetScale(_SetScale ev, Emitter<_State> emit) {
emit(state.copyWith(scale: ev.scale));
}
void _onSetError(_SetError ev, Emitter<_State> emit) {
_log.info("$ev");
emit(state.copyWith(error: ExceptionEvent(ev.error, ev.stackTrace)));
}
void _onSetMessage(_SetMessage ev, Emitter<_State> emit) {
_log.info("$ev");
emit(state.copyWith(message: ev.message));
}
void _initItemController(Collection collection) {
try {
itemsController = collectionsController.stream.value
.itemsControllerByCollection(collection);
} catch (e) {
_log.info(
"[_initItemController] Collection not found in global controller, building new ad-hoc item controller",
e,
);
itemsController = CollectionItemsController(
_c,
account: account,
collection: collection,
onCollectionUpdated: (_) {},
);
}
}
void _clearSelection(Emitter<_State> emit) {
emit(state.copyWith(
selectedItems: const {},
isSelectionRemovable: true,
isSelectionManageableFile: true,
));
}
/// Convert [CollectionItem] to the corresponding [_Item]
_TransformResult _transformItems(
List<CollectionItem> items, CollectionItemSort sort) {
final sorter = CollectionSorter.fromSortType(sort);
final sortedItems = sorter(items);
final dateHelper = photo_list_util.DateGroupHelper(isMonthOnly: false);
final transformed = <_Item>[];
for (int i = 0; i < sortedItems.length; ++i) {
final item = sortedItems[i];
if (item is CollectionFileItem) {
if (sorter is CollectionTimeSorter &&
_c.pref.isAlbumBrowserShowDateOr()) {
final date = dateHelper.onFile(item.file);
if (date != null) {
transformed.add(_DateItem(date: date));
}
}
if (file_util.isSupportedImageMime(item.file.fdMime ?? "")) {
transformed.add(_PhotoItem(
original: item,
file: item.file,
account: account,
));
} else if (file_util.isSupportedVideoMime(item.file.fdMime ?? "")) {
transformed.add(_VideoItem(
original: item,
file: item.file,
account: account,
));
} else {
_log.shout(
"[_transformItems] Unsupported file format: ${item.file.fdMime}");
}
} else if (item is CollectionLabelItem) {
if (state.isEditMode) {
transformed.add(_EditLabelListItem(
original: item,
id: item.id,
text: item.text,
onEditPressed: () {
// TODO
},
));
} else {
transformed.add(_LabelItem(
original: item,
id: item.id,
text: item.text,
));
}
}
}
return _TransformResult(
sorted: sortedItems,
transformed: transformed,
);
}
2023-04-17 18:15:29 +02:00
String? _getCoverUrlByItems() {
try {
final firstFile =
2023-04-17 18:15:29 +02:00
state.transformedItems.whereType<_FileItem>().first.file;
return api_util.getFilePreviewUrlByFileId(
account,
firstFile.fdId,
width: k.coverSize,
height: k.coverSize,
isKeepAspectRatio: false,
);
} catch (_) {
return null;
}
}
static String? _getCoverUrl(Collection collection) {
try {
return collection.contentProvider.getCoverUrl(k.coverSize, k.coverSize);
} catch (_) {
return null;
}
}
2023-04-17 18:15:29 +02:00
void _updateCover(Emitter<_State> emit) {
var coverUrl = _getCoverUrl(state.collection);
coverUrl ??= _getCoverUrlByItems();
if (coverUrl != state.coverUrl) {
emit(state.copyWith(coverUrl: coverUrl));
}
}
final DiContainer _c;
final Account account;
final PrefController prefController;
final CollectionsController collectionsController;
late final CollectionItemsController itemsController;
/// Specify if the supplied [collection] is an "inline" one, which means it's
/// not returned from the collection controller but rather created temporarily
final bool _isAdHocCollection;
StreamSubscription? _collectionControllerSubscription;
StreamSubscription? _itemsControllerSubscription;
2023-05-20 13:07:07 +02:00
var _isHandlingError = false;
}
class _TransformResult {
const _TransformResult({
required this.sorted,
required this.transformed,
});
final List<CollectionItem> sorted;
final List<_Item> transformed;
}