2023-04-13 17:32:31 +02:00
|
|
|
import 'package:collection/collection.dart';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
2023-05-01 19:05:33 +02:00
|
|
|
import 'package:nc_photos/debug_util.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:nc_photos/di_container.dart';
|
2023-04-17 18:15:29 +02:00
|
|
|
import 'package:nc_photos/entity/album/cover_provider.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:nc_photos/entity/album/item.dart';
|
|
|
|
import 'package:nc_photos/entity/album/provider.dart';
|
|
|
|
import 'package:nc_photos/entity/collection.dart';
|
|
|
|
import 'package:nc_photos/entity/collection/adapter.dart';
|
|
|
|
import 'package:nc_photos/entity/collection/builder.dart';
|
|
|
|
import 'package:nc_photos/entity/collection/content_provider/album.dart';
|
2023-05-01 19:05:33 +02:00
|
|
|
import 'package:nc_photos/entity/collection/util.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:nc_photos/entity/collection_item.dart';
|
|
|
|
import 'package:nc_photos/entity/collection_item/album_item_adapter.dart';
|
|
|
|
import 'package:nc_photos/entity/collection_item/new_item.dart';
|
|
|
|
import 'package:nc_photos/entity/collection_item/util.dart';
|
|
|
|
import 'package:nc_photos/entity/file.dart';
|
|
|
|
import 'package:nc_photos/entity/file_descriptor.dart';
|
2023-05-01 19:05:33 +02:00
|
|
|
import 'package:nc_photos/entity/sharee.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:nc_photos/iterable_extension.dart';
|
|
|
|
import 'package:nc_photos/object_extension.dart';
|
2023-04-17 18:15:29 +02:00
|
|
|
import 'package:nc_photos/or_null.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:nc_photos/use_case/album/add_file_to_album.dart';
|
|
|
|
import 'package:nc_photos/use_case/album/edit_album.dart';
|
|
|
|
import 'package:nc_photos/use_case/album/remove_album.dart';
|
|
|
|
import 'package:nc_photos/use_case/album/remove_from_album.dart';
|
2023-05-01 19:05:33 +02:00
|
|
|
import 'package:nc_photos/use_case/album/share_album_with_user.dart';
|
|
|
|
import 'package:nc_photos/use_case/album/unshare_album_with_user.dart';
|
2023-05-15 15:23:27 +02:00
|
|
|
import 'package:nc_photos/use_case/import_pending_shared_album.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:nc_photos/use_case/preprocess_album.dart';
|
2023-05-13 14:45:35 +02:00
|
|
|
import 'package:nc_photos/use_case/unimport_shared_album.dart';
|
2023-04-22 05:43:13 +02:00
|
|
|
import 'package:nc_photos/use_case/update_album_with_actual_items.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:np_codegen/np_codegen.dart';
|
2023-05-01 19:05:33 +02:00
|
|
|
import 'package:np_common/ci_string.dart';
|
2023-04-13 17:32:31 +02:00
|
|
|
import 'package:np_common/type.dart';
|
|
|
|
import 'package:tuple/tuple.dart';
|
|
|
|
|
|
|
|
part 'album.g.dart';
|
|
|
|
|
|
|
|
@npLog
|
|
|
|
class CollectionAlbumAdapter implements CollectionAdapter {
|
|
|
|
CollectionAlbumAdapter(this._c, this.account, this.collection)
|
|
|
|
: assert(require(_c)),
|
|
|
|
_provider = collection.contentProvider as CollectionAlbumProvider;
|
|
|
|
|
|
|
|
static bool require(DiContainer c) => PreProcessAlbum.require(c);
|
|
|
|
|
|
|
|
@override
|
|
|
|
Stream<List<CollectionItem>> listItem() async* {
|
|
|
|
final items = await PreProcessAlbum(_c)(account, _provider.album);
|
|
|
|
yield items.map<CollectionItem>((i) {
|
|
|
|
if (i is AlbumFileItem) {
|
|
|
|
return CollectionFileItemAlbumAdapter(i);
|
|
|
|
} else if (i is AlbumLabelItem) {
|
|
|
|
return CollectionLabelItemAlbumAdapter(i);
|
|
|
|
} else {
|
|
|
|
_log.shout("[listItem] Unknown item type: ${i.runtimeType}");
|
|
|
|
throw UnimplementedError("Unknown item type: ${i.runtimeType}");
|
|
|
|
}
|
|
|
|
}).toList();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<int> addFiles(
|
|
|
|
List<FileDescriptor> files, {
|
|
|
|
ErrorWithValueHandler<FileDescriptor>? onError,
|
|
|
|
required ValueChanged<Collection> onCollectionUpdated,
|
|
|
|
}) async {
|
|
|
|
try {
|
|
|
|
final newAlbum =
|
|
|
|
await AddFileToAlbum(_c)(account, _provider.album, files);
|
|
|
|
onCollectionUpdated(CollectionBuilder.byAlbum(account, newAlbum));
|
|
|
|
return files.length;
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
for (final f in files) {
|
|
|
|
onError?.call(f, e, stackTrace);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<Collection> edit({
|
|
|
|
String? name,
|
|
|
|
List<CollectionItem>? items,
|
|
|
|
CollectionItemSort? itemSort,
|
2023-04-17 18:15:29 +02:00
|
|
|
OrNull<FileDescriptor>? cover,
|
2023-04-22 16:32:34 +02:00
|
|
|
List<CollectionItem>? knownItems,
|
2023-04-13 17:32:31 +02:00
|
|
|
}) async {
|
2023-04-17 18:15:29 +02:00
|
|
|
assert(name != null || items != null || itemSort != null || cover != null);
|
2023-04-13 17:32:31 +02:00
|
|
|
final newItems = items?.run((items) => items
|
|
|
|
.map((e) {
|
|
|
|
if (e is AlbumAdaptedCollectionItem) {
|
|
|
|
return e.albumItem;
|
|
|
|
} else if (e is NewCollectionLabelItem) {
|
|
|
|
// new labels
|
|
|
|
return AlbumLabelItem(
|
|
|
|
addedBy: account.userId,
|
|
|
|
addedAt: e.createdAt,
|
|
|
|
text: e.text,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
_log.severe("[edit] Unsupported type: ${e.runtimeType}");
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.whereNotNull()
|
|
|
|
.toList());
|
|
|
|
final newAlbum = await EditAlbum(_c)(
|
|
|
|
account,
|
|
|
|
_provider.album,
|
|
|
|
name: name,
|
|
|
|
items: newItems,
|
|
|
|
itemSort: itemSort,
|
2023-04-17 18:15:29 +02:00
|
|
|
cover: cover,
|
2023-04-22 16:32:34 +02:00
|
|
|
knownItems: knownItems
|
|
|
|
?.whereType<AlbumAdaptedCollectionItem>()
|
|
|
|
.map((e) => e.albumItem)
|
|
|
|
.toList(),
|
2023-04-13 17:32:31 +02:00
|
|
|
);
|
|
|
|
return collection.copyWith(
|
|
|
|
name: name,
|
|
|
|
contentProvider: _provider.copyWith(album: newAlbum),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<int> removeItems(
|
|
|
|
List<CollectionItem> items, {
|
|
|
|
ErrorWithValueIndexedHandler<CollectionItem>? onError,
|
|
|
|
required ValueChanged<Collection> onCollectionUpdated,
|
|
|
|
}) async {
|
|
|
|
try {
|
|
|
|
final group = items
|
|
|
|
.withIndex()
|
|
|
|
.groupListsBy((e) => e.item2 is AlbumAdaptedCollectionItem);
|
|
|
|
var failed = 0;
|
|
|
|
if (group[true]?.isNotEmpty ?? false) {
|
|
|
|
final newAlbum = await RemoveFromAlbum(_c)(
|
|
|
|
account,
|
|
|
|
_provider.album,
|
|
|
|
group[true]!
|
|
|
|
.map((e) => e.item2)
|
|
|
|
.cast<AlbumAdaptedCollectionItem>()
|
|
|
|
.map((e) => e.albumItem)
|
|
|
|
.toList(),
|
|
|
|
onError: (i, item, e, stackTrace) {
|
|
|
|
++failed;
|
|
|
|
final actualIndex = group[true]![i].item1;
|
|
|
|
try {
|
|
|
|
onError?.call(actualIndex, items[actualIndex], e, stackTrace);
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.severe("[removeItems] Unknown error", e, stackTrace);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
onCollectionUpdated(collection.copyWith(
|
|
|
|
contentProvider: _provider.copyWith(
|
|
|
|
album: newAlbum,
|
|
|
|
),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
for (final pair in (group[false] ?? const <Tuple2<int, int>>[])) {
|
|
|
|
final actualIndex = pair.item1;
|
|
|
|
onError?.call(
|
|
|
|
actualIndex,
|
|
|
|
items[actualIndex],
|
|
|
|
UnsupportedError(
|
|
|
|
"Unsupported item type: ${items[actualIndex].runtimeType}"),
|
|
|
|
StackTrace.current,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return (group[true] ?? []).length - failed;
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
for (final pair in items.withIndex()) {
|
|
|
|
onError?.call(pair.item1, pair.item2, e, stackTrace);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-01 19:05:33 +02:00
|
|
|
@override
|
|
|
|
Future<CollectionShareResult> share(
|
|
|
|
Sharee sharee, {
|
|
|
|
required ValueChanged<Collection> onCollectionUpdated,
|
|
|
|
}) async {
|
|
|
|
var fileFailed = false;
|
|
|
|
final newAlbum = await ShareAlbumWithUser(_c.shareRepo, _c.albumRepo)(
|
|
|
|
account,
|
|
|
|
_provider.album,
|
|
|
|
sharee,
|
|
|
|
onShareFileFailed: (f, e, stackTrace) {
|
|
|
|
_log.severe("[share] Failed to share file: ${logFilename(f.path)}", e,
|
|
|
|
stackTrace);
|
|
|
|
fileFailed = true;
|
|
|
|
},
|
|
|
|
);
|
|
|
|
onCollectionUpdated(CollectionBuilder.byAlbum(account, newAlbum));
|
|
|
|
return fileFailed
|
|
|
|
? CollectionShareResult.partial
|
|
|
|
: CollectionShareResult.ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Future<CollectionShareResult> unshare(
|
|
|
|
CiString userId, {
|
|
|
|
required ValueChanged<Collection> onCollectionUpdated,
|
|
|
|
}) async {
|
|
|
|
var fileFailed = false;
|
|
|
|
final newAlbum = await UnshareAlbumWithUser(_c)(
|
|
|
|
account,
|
|
|
|
_provider.album,
|
|
|
|
userId,
|
|
|
|
onUnshareFileFailed: (f, e, stackTrace) {
|
|
|
|
_log.severe("[unshare] Failed to unshare file: ${logFilename(f.path)}",
|
|
|
|
e, stackTrace);
|
|
|
|
fileFailed = true;
|
|
|
|
},
|
|
|
|
);
|
|
|
|
onCollectionUpdated(CollectionBuilder.byAlbum(account, newAlbum));
|
|
|
|
return fileFailed
|
|
|
|
? CollectionShareResult.partial
|
|
|
|
: CollectionShareResult.ok;
|
|
|
|
}
|
|
|
|
|
2023-05-15 15:23:27 +02:00
|
|
|
@override
|
|
|
|
Future<Collection> importPendingShared() async {
|
|
|
|
final newAlbum =
|
|
|
|
await ImportPendingSharedAlbum(_c)(account, _provider.album);
|
|
|
|
return CollectionBuilder.byAlbum(account, newAlbum);
|
|
|
|
}
|
|
|
|
|
2023-04-13 17:32:31 +02:00
|
|
|
@override
|
|
|
|
Future<CollectionItem> adaptToNewItem(NewCollectionItem original) async {
|
|
|
|
if (original is NewCollectionFileItem) {
|
|
|
|
final item = AlbumStaticProvider.of(_provider.album)
|
|
|
|
.items
|
|
|
|
.whereType<AlbumFileItem>()
|
|
|
|
.firstWhere((e) => e.file.compareServerIdentity(original.file));
|
|
|
|
return CollectionFileItemAlbumAdapter(item);
|
|
|
|
} else if (original is NewCollectionLabelItem) {
|
|
|
|
final item = AlbumStaticProvider.of(_provider.album)
|
|
|
|
.items
|
|
|
|
.whereType<AlbumLabelItem>()
|
|
|
|
.sorted((a, b) => a.addedAt.compareTo(b.addedAt))
|
|
|
|
.reversed
|
|
|
|
.firstWhere((e) => e.text == original.text);
|
|
|
|
return CollectionLabelItemAlbumAdapter(item);
|
|
|
|
} else {
|
|
|
|
throw UnsupportedError("Unsupported type: ${original.runtimeType}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2023-04-17 18:15:29 +02:00
|
|
|
bool isItemRemovable(CollectionItem item) {
|
|
|
|
if (_provider.album.provider is! AlbumStaticProvider) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (_provider.album.albumFile?.isOwned(account.userId) == true) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (item is! AlbumAdaptedCollectionItem) {
|
|
|
|
_log.warning("[isItemRemovable] Unknown item type: ${item.runtimeType}");
|
2023-04-13 17:32:31 +02:00
|
|
|
return true;
|
|
|
|
}
|
2023-04-17 18:15:29 +02:00
|
|
|
return item.albumItem.addedBy == account.userId;
|
2023-04-13 17:32:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2023-05-13 14:45:35 +02:00
|
|
|
Future<void> remove() {
|
|
|
|
if (_provider.album.albumFile?.isOwned(account.userId) == true) {
|
|
|
|
return RemoveAlbum(_c)(account, _provider.album);
|
|
|
|
} else {
|
|
|
|
return UnimportSharedAlbum(_c)(account, _provider.album);
|
|
|
|
}
|
|
|
|
}
|
2023-04-13 17:32:31 +02:00
|
|
|
|
2023-04-17 18:15:29 +02:00
|
|
|
@override
|
|
|
|
bool isPermitted(CollectionCapability capability) {
|
|
|
|
if (!_provider.capabilities.contains(capability)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (_provider.album.albumFile?.isOwned(account.userId) == true) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return _provider.guestCapabilities.contains(capability);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool isManualCover() =>
|
|
|
|
_provider.album.coverProvider is AlbumManualCoverProvider;
|
|
|
|
|
2023-04-22 05:43:13 +02:00
|
|
|
@override
|
|
|
|
Future<Collection?> updatePostLoad(List<CollectionItem> items) async {
|
|
|
|
final album = await UpdateAlbumWithActualItems(_c.albumRepo)(
|
|
|
|
account,
|
|
|
|
_provider.album,
|
|
|
|
items
|
|
|
|
.whereType<AlbumAdaptedCollectionItem>()
|
|
|
|
.map((e) => e.albumItem)
|
|
|
|
.toList(),
|
|
|
|
);
|
|
|
|
if (!identical(album, _provider.album)) {
|
|
|
|
return CollectionBuilder.byAlbum(account, album);
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-13 17:32:31 +02:00
|
|
|
final DiContainer _c;
|
|
|
|
final Account account;
|
|
|
|
final Collection collection;
|
|
|
|
|
|
|
|
final CollectionAlbumProvider _provider;
|
|
|
|
}
|