mirror of
https://gitlab.com/nkming2/nc-photos.git
synced 2025-01-23 09:16:19 +01:00
579 lines
17 KiB
Dart
579 lines
17 KiB
Dart
part of '../collection_browser.dart';
|
|
|
|
@npLog
|
|
class _Bloc extends Bloc<_Event, _State> with BlocLogger {
|
|
_Bloc({
|
|
required DiContainer container,
|
|
required this.account,
|
|
required this.prefController,
|
|
required this.collectionsController,
|
|
required this.filesController,
|
|
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.albumBrowserZoomLevelValue,
|
|
)) {
|
|
_initItemController(collection);
|
|
|
|
on<_UpdateCollection>(_onUpdateCollection);
|
|
on<_LoadItems>(_onLoad);
|
|
on<_TransformItems>(_onTransformItems);
|
|
on<_ImportPendingSharedCollection>(_onImportPendingSharedCollection);
|
|
|
|
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);
|
|
|
|
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) {
|
|
_subscriptions.add(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");
|
|
}
|
|
}
|
|
|
|
@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) {
|
|
return currentState.scale == nextState.scale;
|
|
};
|
|
|
|
@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);
|
|
}
|
|
|
|
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));
|
|
_updateCover(emit);
|
|
}
|
|
|
|
Future<void> _onLoad(_LoadItems ev, Emitter<_State> emit) async {
|
|
_log.info(ev);
|
|
await Future.wait([
|
|
emit.forEach<CollectionItemStreamData>(
|
|
itemsController.stream,
|
|
onData: (data) => state.copyWith(
|
|
items: _filterItems(data.items, state.itemsWhitelist),
|
|
rawItems: data.items,
|
|
isLoading: data.hasNext,
|
|
),
|
|
onError: (e, stackTrace) {
|
|
_log.severe("[_onLoad] Uncaught exception", e, stackTrace);
|
|
return state.copyWith(
|
|
isLoading: false,
|
|
error: ExceptionEvent(e, stackTrace),
|
|
);
|
|
},
|
|
),
|
|
emit.forEach<FilesStreamEvent>(
|
|
filesController.stream,
|
|
onData: (data) {
|
|
final whitelist = HashSet.of(data.dataMap.keys);
|
|
return state.copyWith(
|
|
items: _filterItems(state.rawItems, whitelist),
|
|
itemsWhitelist: whitelist,
|
|
);
|
|
},
|
|
onError: (e, stackTrace) {
|
|
_log.severe("[_onLoad] Uncaught exception", e, stackTrace);
|
|
return state.copyWith(
|
|
isLoading: false,
|
|
error: ExceptionEvent(e, stackTrace),
|
|
);
|
|
},
|
|
),
|
|
]);
|
|
}
|
|
|
|
void _onTransformItems(_TransformItems ev, Emitter<_State> emit) {
|
|
_log.info(ev);
|
|
final result = _transformItems(ev.items, state.collection.itemSort);
|
|
emit(state.copyWith(transformedItems: result.transformed));
|
|
_updateCover(emit);
|
|
}
|
|
|
|
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);
|
|
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);
|
|
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,
|
|
));
|
|
}
|
|
|
|
void _onUnsetCover(_UnsetCover ev, Emitter<_State> emit) {
|
|
_log.info(ev);
|
|
collectionsController.edit(state.collection, cover: const OrNull(null));
|
|
}
|
|
|
|
void _onSetSelectedItems(_SetSelectedItems ev, Emitter<_State> emit) {
|
|
_log.info(ev);
|
|
final adapter = CollectionAdapter.of(_c, account, state.collection);
|
|
emit(state.copyWith(
|
|
selectedItems: ev.items,
|
|
isSelectionRemovable: ev.items
|
|
.whereType<_ActualItem>()
|
|
.map((e) => e.original)
|
|
.any(adapter.isItemRemovable),
|
|
isSelectionDeletable: ev.items
|
|
.whereType<_ActualItem>()
|
|
.map((e) => e.original)
|
|
.any(adapter.isItemDeletable),
|
|
));
|
|
}
|
|
|
|
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 adapter = CollectionAdapter.of(_c, account, state.collection);
|
|
final selectedItems = selected
|
|
.whereType<_ActualItem>()
|
|
.map((e) => e.original)
|
|
.where(adapter.isItemRemovable)
|
|
.toList();
|
|
if (selectedItems.isNotEmpty) {
|
|
unawaited(itemsController.removeItems(selectedItems));
|
|
}
|
|
}
|
|
|
|
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) {
|
|
filesController.updateProperty(
|
|
selectedFiles,
|
|
isArchived: const OrNull(true),
|
|
errorBuilder: (fileIds) => _ArchiveFailedError(fileIds.length),
|
|
);
|
|
}
|
|
}
|
|
|
|
Future<void> _onDeleteSelectedItems(
|
|
_DeleteSelectedItems ev, Emitter<_State> emit) async {
|
|
_log.info(ev);
|
|
final selected = state.selectedItems;
|
|
_clearSelection(emit);
|
|
final adapter = CollectionAdapter.of(_c, account, state.collection);
|
|
final selectedItems = selected
|
|
.whereType<_ActualItem>()
|
|
.map((e) => e.original)
|
|
.where(adapter.isItemRemovable)
|
|
.toList();
|
|
final selectedFiles = selected
|
|
.whereType<_FileItem>()
|
|
.where((e) => adapter.isItemDeletable(e.original))
|
|
.map((e) => e.file)
|
|
.toList();
|
|
if (selectedFiles.isNotEmpty) {
|
|
await filesController.remove(
|
|
selectedFiles,
|
|
errorBuilder: (fileIds) {
|
|
return _RemoveFailedError(fileIds.length);
|
|
},
|
|
);
|
|
// deleting files will also remove them from the collection
|
|
unawaited(itemsController.removeItems(selectedItems));
|
|
}
|
|
}
|
|
|
|
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,
|
|
filesController: filesController,
|
|
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,
|
|
);
|
|
}
|
|
|
|
/// Remove file items not recognized by the app
|
|
List<CollectionItem> _filterItems(
|
|
List<CollectionItem> rawItems, Set<int>? whitelist) {
|
|
if (whitelist == null) {
|
|
return rawItems;
|
|
}
|
|
final results = rawItems.where((e) {
|
|
if (e is CollectionFileItem) {
|
|
return whitelist.contains(e.file.fdId);
|
|
} else {
|
|
return true;
|
|
}
|
|
}).toList();
|
|
if (rawItems.length != results.length) {
|
|
_log.fine(
|
|
"[_filterItems] ${rawItems.length - results.length} items filtered out");
|
|
}
|
|
return results;
|
|
}
|
|
|
|
String? _getCoverUrlByItems() {
|
|
try {
|
|
final firstFile =
|
|
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;
|
|
}
|
|
}
|
|
|
|
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;
|
|
final FilesController filesController;
|
|
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;
|
|
|
|
final _subscriptions = <StreamSubscription>[];
|
|
var _isHandlingError = false;
|
|
}
|
|
|
|
class _TransformResult {
|
|
const _TransformResult({
|
|
required this.sorted,
|
|
required this.transformed,
|
|
});
|
|
|
|
final List<CollectionItem> sorted;
|
|
final List<_Item> transformed;
|
|
}
|