mirror of
https://github.com/immich-app/immich.git
synced 2025-01-07 20:36:48 +01:00
3c38851d50
* add native player library * splitup the player * stateful widget * refactor: native_video_player * fix: handle buffering * turn on volume when video plays * fix: aspect ratio * fix: handle remote asset orientation * refinements and fixes fix orientation for remote assets wip separate widget separate video loader widget fixed memory leak optimized seeking, cleanup debug context pop use global key back to one widget fixed rebuild wait for swipe animation to finish smooth hero animation for remote videos faster scroll animation * clean up logging * refactor aspect ratio calculation * removed unnecessary import * transitive dependencies * fixed referencing uninitialized orientation * use correct ref to build android * higher res placeholder for local videos * slightly lower delay * await things * fix controls when swiping between image and video * linting * extra smooth seeking, add comments * chore: generate router page * use current asset provider and loadAsset * fix stack handling * improved motion photo handling * use visibility for motion videos * error handling for async calls * fix duplicate key error * maybe fix duplicate key error * increase delay for hero animation * faster initialization for remote videos * ensure dimensions for memory cards * make aspect ratio logic reusable, optimizations * refactor: move exif search from aspect ratio to orientation * local orientation on ios is unreliable; prefer remote * fix no audio in silent mode on ios * increase bottom bar opacity to account for hdr * remove unused import * fix live photo play button not updating * fix map marker -> galleryviewer * remove video_player * fix hdr playback on android * fix looping * remove unused dependencies * update to latest player commit * fix player controls hiding when video is not playing * fix restart video * stop showing motion video after ending when looping is disabled * delay video initialization to avoid placeholder flicker * faster animation * shorter delay * small delay for image -> video on android * fix: lint * hide stacked children when controls are hidden, avoid bottom bar dropping --------- Co-authored-by: Alex <alex.tran1502@gmail.com> Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com> Co-authored-by: mertalev <101130780+mertalev@users.noreply.github.com>
592 lines
17 KiB
Dart
592 lines
17 KiB
Dart
import 'dart:convert';
|
|
import 'dart:io';
|
|
|
|
import 'package:immich_mobile/entities/exif_info.entity.dart';
|
|
import 'package:immich_mobile/utils/hash.dart';
|
|
import 'package:isar/isar.dart';
|
|
import 'package:openapi/api.dart';
|
|
import 'package:photo_manager/photo_manager.dart' show AssetEntity;
|
|
import 'package:immich_mobile/extensions/string_extensions.dart';
|
|
import 'package:path/path.dart' as p;
|
|
|
|
part 'asset.entity.g.dart';
|
|
|
|
/// Asset (online or local)
|
|
@Collection(inheritance: false)
|
|
class Asset {
|
|
Asset.remote(AssetResponseDto remote)
|
|
: remoteId = remote.id,
|
|
checksum = remote.checksum,
|
|
fileCreatedAt = remote.fileCreatedAt,
|
|
fileModifiedAt = remote.fileModifiedAt,
|
|
updatedAt = remote.updatedAt,
|
|
durationInSeconds = remote.duration.toDuration()?.inSeconds ?? 0,
|
|
type = remote.type.toAssetType(),
|
|
fileName = remote.originalFileName,
|
|
height = remote.exifInfo?.exifImageHeight?.toInt(),
|
|
width = remote.exifInfo?.exifImageWidth?.toInt(),
|
|
livePhotoVideoId = remote.livePhotoVideoId,
|
|
ownerId = fastHash(remote.ownerId),
|
|
exifInfo =
|
|
remote.exifInfo != null ? ExifInfo.fromDto(remote.exifInfo!) : null,
|
|
isFavorite = remote.isFavorite,
|
|
isArchived = remote.isArchived,
|
|
isTrashed = remote.isTrashed,
|
|
isOffline = remote.isOffline,
|
|
// workaround to nullify stackPrimaryAssetId for the parent asset until we refactor the mobile app
|
|
// stack handling to properly handle it
|
|
stackPrimaryAssetId = remote.stack?.primaryAssetId == remote.id
|
|
? null
|
|
: remote.stack?.primaryAssetId,
|
|
stackCount = remote.stack?.assetCount ?? 0,
|
|
stackId = remote.stack?.id,
|
|
thumbhash = remote.thumbhash;
|
|
|
|
Asset({
|
|
this.id = Isar.autoIncrement,
|
|
required this.checksum,
|
|
this.remoteId,
|
|
required this.localId,
|
|
required this.ownerId,
|
|
required this.fileCreatedAt,
|
|
required this.fileModifiedAt,
|
|
required this.updatedAt,
|
|
required this.durationInSeconds,
|
|
required this.type,
|
|
this.width,
|
|
this.height,
|
|
required this.fileName,
|
|
this.livePhotoVideoId,
|
|
this.exifInfo,
|
|
this.isFavorite = false,
|
|
this.isArchived = false,
|
|
this.isTrashed = false,
|
|
this.stackId,
|
|
this.stackPrimaryAssetId,
|
|
this.stackCount = 0,
|
|
this.isOffline = false,
|
|
this.thumbhash,
|
|
});
|
|
|
|
@ignore
|
|
AssetEntity? _local;
|
|
|
|
@ignore
|
|
AssetEntity? get local {
|
|
if (isLocal && _local == null) {
|
|
_local = AssetEntity(
|
|
id: localId!,
|
|
typeInt: isImage ? 1 : 2,
|
|
width: width ?? 0,
|
|
height: height ?? 0,
|
|
duration: durationInSeconds,
|
|
createDateSecond: fileCreatedAt.millisecondsSinceEpoch ~/ 1000,
|
|
modifiedDateSecond: fileModifiedAt.millisecondsSinceEpoch ~/ 1000,
|
|
title: fileName,
|
|
);
|
|
}
|
|
return _local;
|
|
}
|
|
|
|
set local(AssetEntity? assetEntity) => _local = assetEntity;
|
|
|
|
@ignore
|
|
bool _didUpdateLocal = false;
|
|
|
|
@ignore
|
|
Future<AssetEntity> get localAsync async {
|
|
final local = this.local;
|
|
if (local == null) {
|
|
throw Exception('Asset $fileName has no local data');
|
|
}
|
|
|
|
final updatedLocal =
|
|
_didUpdateLocal ? local : await local.obtainForNewProperties();
|
|
if (updatedLocal == null) {
|
|
throw Exception('Could not fetch local data for $fileName');
|
|
}
|
|
|
|
this.local = updatedLocal;
|
|
_didUpdateLocal = true;
|
|
return updatedLocal;
|
|
}
|
|
|
|
Id id = Isar.autoIncrement;
|
|
|
|
/// stores the raw SHA1 bytes as a base64 String
|
|
/// because Isar cannot sort lists of byte arrays
|
|
String checksum;
|
|
|
|
String? thumbhash;
|
|
|
|
@Index(unique: false, replace: false, type: IndexType.hash)
|
|
String? remoteId;
|
|
|
|
@Index(unique: false, replace: false, type: IndexType.hash)
|
|
String? localId;
|
|
|
|
@Index(
|
|
unique: true,
|
|
replace: false,
|
|
composite: [CompositeIndex("checksum", type: IndexType.hash)],
|
|
)
|
|
int ownerId;
|
|
|
|
DateTime fileCreatedAt;
|
|
|
|
DateTime fileModifiedAt;
|
|
|
|
DateTime updatedAt;
|
|
|
|
int durationInSeconds;
|
|
|
|
@Enumerated(EnumType.ordinal)
|
|
AssetType type;
|
|
|
|
short? width;
|
|
|
|
short? height;
|
|
|
|
String fileName;
|
|
|
|
String? livePhotoVideoId;
|
|
|
|
bool isFavorite;
|
|
|
|
bool isArchived;
|
|
|
|
bool isTrashed;
|
|
|
|
bool isOffline;
|
|
|
|
@ignore
|
|
ExifInfo? exifInfo;
|
|
|
|
String? stackId;
|
|
|
|
String? stackPrimaryAssetId;
|
|
|
|
int stackCount;
|
|
|
|
/// Returns null if the asset has no sync access to the exif info
|
|
@ignore
|
|
double? get aspectRatio {
|
|
final orientatedWidth = this.orientatedWidth;
|
|
final orientatedHeight = this.orientatedHeight;
|
|
|
|
if (orientatedWidth != null &&
|
|
orientatedHeight != null &&
|
|
orientatedWidth > 0 &&
|
|
orientatedHeight > 0) {
|
|
return orientatedWidth.toDouble() / orientatedHeight.toDouble();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/// `true` if this [Asset] is present on the device
|
|
@ignore
|
|
bool get isLocal => localId != null;
|
|
|
|
@ignore
|
|
bool get isInDb => id != Isar.autoIncrement;
|
|
|
|
@ignore
|
|
String get name => p.withoutExtension(fileName);
|
|
|
|
/// `true` if this [Asset] is present on the server
|
|
@ignore
|
|
bool get isRemote => remoteId != null;
|
|
|
|
@ignore
|
|
bool get isImage => type == AssetType.image;
|
|
|
|
@ignore
|
|
bool get isVideo => type == AssetType.video;
|
|
|
|
@ignore
|
|
bool get isMotionPhoto => livePhotoVideoId != null;
|
|
|
|
@ignore
|
|
AssetState get storage {
|
|
if (isRemote && isLocal) {
|
|
return AssetState.merged;
|
|
} else if (isRemote) {
|
|
return AssetState.remote;
|
|
} else if (isLocal) {
|
|
return AssetState.local;
|
|
} else {
|
|
throw Exception("Asset has illegal state: $this");
|
|
}
|
|
}
|
|
|
|
@ignore
|
|
Duration get duration => Duration(seconds: durationInSeconds);
|
|
|
|
// ignore: invalid_annotation_target
|
|
@ignore
|
|
set byteHash(List<int> hash) => checksum = base64.encode(hash);
|
|
|
|
/// Returns null if the asset has no sync access to the exif info
|
|
@ignore
|
|
@pragma('vm:prefer-inline')
|
|
bool? get isFlipped {
|
|
final exifInfo = this.exifInfo;
|
|
if (exifInfo != null) {
|
|
return exifInfo.isFlipped;
|
|
}
|
|
|
|
if (_didUpdateLocal && Platform.isAndroid) {
|
|
final local = this.local;
|
|
if (local == null) {
|
|
throw Exception('Asset $fileName has no local data');
|
|
}
|
|
return local.orientation == 90 || local.orientation == 270;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/// Returns null if the asset has no sync access to the exif info
|
|
@ignore
|
|
@pragma('vm:prefer-inline')
|
|
int? get orientatedHeight {
|
|
final isFlipped = this.isFlipped;
|
|
if (isFlipped == null) {
|
|
return null;
|
|
}
|
|
|
|
return isFlipped ? width : height;
|
|
}
|
|
|
|
/// Returns null if the asset has no sync access to the exif info
|
|
@ignore
|
|
@pragma('vm:prefer-inline')
|
|
int? get orientatedWidth {
|
|
final isFlipped = this.isFlipped;
|
|
if (isFlipped == null) {
|
|
return null;
|
|
}
|
|
|
|
return isFlipped ? height : width;
|
|
}
|
|
|
|
@override
|
|
bool operator ==(other) {
|
|
if (other is! Asset) return false;
|
|
if (identical(this, other)) return true;
|
|
return id == other.id &&
|
|
checksum == other.checksum &&
|
|
remoteId == other.remoteId &&
|
|
localId == other.localId &&
|
|
ownerId == other.ownerId &&
|
|
fileCreatedAt.isAtSameMomentAs(other.fileCreatedAt) &&
|
|
fileModifiedAt.isAtSameMomentAs(other.fileModifiedAt) &&
|
|
updatedAt.isAtSameMomentAs(other.updatedAt) &&
|
|
durationInSeconds == other.durationInSeconds &&
|
|
type == other.type &&
|
|
width == other.width &&
|
|
height == other.height &&
|
|
fileName == other.fileName &&
|
|
livePhotoVideoId == other.livePhotoVideoId &&
|
|
isFavorite == other.isFavorite &&
|
|
isLocal == other.isLocal &&
|
|
isArchived == other.isArchived &&
|
|
isTrashed == other.isTrashed &&
|
|
stackCount == other.stackCount &&
|
|
stackPrimaryAssetId == other.stackPrimaryAssetId &&
|
|
stackId == other.stackId;
|
|
}
|
|
|
|
@override
|
|
@ignore
|
|
int get hashCode =>
|
|
id.hashCode ^
|
|
checksum.hashCode ^
|
|
remoteId.hashCode ^
|
|
localId.hashCode ^
|
|
ownerId.hashCode ^
|
|
fileCreatedAt.hashCode ^
|
|
fileModifiedAt.hashCode ^
|
|
updatedAt.hashCode ^
|
|
durationInSeconds.hashCode ^
|
|
type.hashCode ^
|
|
width.hashCode ^
|
|
height.hashCode ^
|
|
fileName.hashCode ^
|
|
livePhotoVideoId.hashCode ^
|
|
isFavorite.hashCode ^
|
|
isLocal.hashCode ^
|
|
isArchived.hashCode ^
|
|
isTrashed.hashCode ^
|
|
stackCount.hashCode ^
|
|
stackPrimaryAssetId.hashCode ^
|
|
stackId.hashCode;
|
|
|
|
/// Returns `true` if this [Asset] can updated with values from parameter [a]
|
|
bool canUpdate(Asset a) {
|
|
assert(isInDb);
|
|
assert(checksum == a.checksum);
|
|
assert(a.storage != AssetState.merged);
|
|
return a.updatedAt.isAfter(updatedAt) ||
|
|
a.isRemote && !isRemote ||
|
|
a.isLocal && !isLocal ||
|
|
width == null && a.width != null ||
|
|
height == null && a.height != null ||
|
|
livePhotoVideoId == null && a.livePhotoVideoId != null ||
|
|
isFavorite != a.isFavorite ||
|
|
isArchived != a.isArchived ||
|
|
isTrashed != a.isTrashed ||
|
|
isOffline != a.isOffline ||
|
|
a.exifInfo?.latitude != exifInfo?.latitude ||
|
|
a.exifInfo?.longitude != exifInfo?.longitude ||
|
|
// no local stack count or different count from remote
|
|
a.thumbhash != thumbhash ||
|
|
stackId != a.stackId ||
|
|
stackCount != a.stackCount ||
|
|
stackPrimaryAssetId == null && a.stackPrimaryAssetId != null;
|
|
}
|
|
|
|
/// Returns a new [Asset] with values from this and merged & updated with [a]
|
|
Asset updatedCopy(Asset a) {
|
|
assert(canUpdate(a));
|
|
if (a.updatedAt.isAfter(updatedAt)) {
|
|
// take most values from newer asset
|
|
// keep vales that can never be set by the asset not in DB
|
|
if (a.isRemote) {
|
|
return a._copyWith(
|
|
id: id,
|
|
localId: localId,
|
|
width: a.width ?? width,
|
|
height: a.height ?? height,
|
|
exifInfo: a.exifInfo?.copyWith(id: id) ?? exifInfo,
|
|
);
|
|
} else if (isRemote) {
|
|
return _copyWith(
|
|
localId: localId ?? a.localId,
|
|
width: width ?? a.width,
|
|
height: height ?? a.height,
|
|
exifInfo: exifInfo ?? a.exifInfo?.copyWith(id: id),
|
|
);
|
|
} else {
|
|
// TODO: Revisit this and remove all bool field assignments
|
|
return a._copyWith(
|
|
id: id,
|
|
remoteId: remoteId,
|
|
livePhotoVideoId: livePhotoVideoId,
|
|
// workaround to nullify stackPrimaryAssetId for the parent asset until we refactor the mobile app
|
|
// stack handling to properly handle it
|
|
stackId: stackId,
|
|
stackPrimaryAssetId:
|
|
stackPrimaryAssetId == remoteId ? null : stackPrimaryAssetId,
|
|
stackCount: stackCount,
|
|
isFavorite: isFavorite,
|
|
isArchived: isArchived,
|
|
isTrashed: isTrashed,
|
|
isOffline: isOffline,
|
|
);
|
|
}
|
|
} else {
|
|
// fill in potentially missing values, i.e. merge assets
|
|
if (a.isRemote) {
|
|
// values from remote take precedence
|
|
return _copyWith(
|
|
remoteId: a.remoteId,
|
|
width: a.width,
|
|
height: a.height,
|
|
livePhotoVideoId: a.livePhotoVideoId,
|
|
// workaround to nullify stackPrimaryAssetId for the parent asset until we refactor the mobile app
|
|
// stack handling to properly handle it
|
|
stackId: a.stackId,
|
|
stackPrimaryAssetId: a.stackPrimaryAssetId == a.remoteId
|
|
? null
|
|
: a.stackPrimaryAssetId,
|
|
stackCount: a.stackCount,
|
|
// isFavorite + isArchived are not set by device-only assets
|
|
isFavorite: a.isFavorite,
|
|
isArchived: a.isArchived,
|
|
isTrashed: a.isTrashed,
|
|
isOffline: a.isOffline,
|
|
exifInfo: a.exifInfo?.copyWith(id: id) ?? exifInfo,
|
|
thumbhash: a.thumbhash,
|
|
);
|
|
} else {
|
|
// add only missing values (and set isLocal to true)
|
|
return _copyWith(
|
|
localId: localId ?? a.localId,
|
|
width: width ?? a.width,
|
|
height: height ?? a.height,
|
|
exifInfo: exifInfo ?? a.exifInfo?.copyWith(id: id),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
Asset _copyWith({
|
|
Id? id,
|
|
String? checksum,
|
|
String? remoteId,
|
|
String? localId,
|
|
int? ownerId,
|
|
DateTime? fileCreatedAt,
|
|
DateTime? fileModifiedAt,
|
|
DateTime? updatedAt,
|
|
int? durationInSeconds,
|
|
AssetType? type,
|
|
short? width,
|
|
short? height,
|
|
String? fileName,
|
|
String? livePhotoVideoId,
|
|
bool? isFavorite,
|
|
bool? isArchived,
|
|
bool? isTrashed,
|
|
bool? isOffline,
|
|
ExifInfo? exifInfo,
|
|
String? stackId,
|
|
String? stackPrimaryAssetId,
|
|
int? stackCount,
|
|
String? thumbhash,
|
|
}) =>
|
|
Asset(
|
|
id: id ?? this.id,
|
|
checksum: checksum ?? this.checksum,
|
|
remoteId: remoteId ?? this.remoteId,
|
|
localId: localId ?? this.localId,
|
|
ownerId: ownerId ?? this.ownerId,
|
|
fileCreatedAt: fileCreatedAt ?? this.fileCreatedAt,
|
|
fileModifiedAt: fileModifiedAt ?? this.fileModifiedAt,
|
|
updatedAt: updatedAt ?? this.updatedAt,
|
|
durationInSeconds: durationInSeconds ?? this.durationInSeconds,
|
|
type: type ?? this.type,
|
|
width: width ?? this.width,
|
|
height: height ?? this.height,
|
|
fileName: fileName ?? this.fileName,
|
|
livePhotoVideoId: livePhotoVideoId ?? this.livePhotoVideoId,
|
|
isFavorite: isFavorite ?? this.isFavorite,
|
|
isArchived: isArchived ?? this.isArchived,
|
|
isTrashed: isTrashed ?? this.isTrashed,
|
|
isOffline: isOffline ?? this.isOffline,
|
|
exifInfo: exifInfo ?? this.exifInfo,
|
|
stackId: stackId ?? this.stackId,
|
|
stackPrimaryAssetId: stackPrimaryAssetId ?? this.stackPrimaryAssetId,
|
|
stackCount: stackCount ?? this.stackCount,
|
|
thumbhash: thumbhash ?? this.thumbhash,
|
|
);
|
|
|
|
Future<void> put(Isar db) async {
|
|
await db.assets.put(this);
|
|
if (exifInfo != null) {
|
|
exifInfo!.id = id;
|
|
await db.exifInfos.put(exifInfo!);
|
|
}
|
|
}
|
|
|
|
static int compareById(Asset a, Asset b) => a.id.compareTo(b.id);
|
|
|
|
static int compareByChecksum(Asset a, Asset b) =>
|
|
a.checksum.compareTo(b.checksum);
|
|
|
|
static int compareByOwnerChecksum(Asset a, Asset b) {
|
|
final int ownerIdOrder = a.ownerId.compareTo(b.ownerId);
|
|
if (ownerIdOrder != 0) return ownerIdOrder;
|
|
return compareByChecksum(a, b);
|
|
}
|
|
|
|
static int compareByOwnerChecksumCreatedModified(
|
|
Asset a,
|
|
Asset b,
|
|
) {
|
|
final int ownerIdOrder = a.ownerId.compareTo(b.ownerId);
|
|
if (ownerIdOrder != 0) return ownerIdOrder;
|
|
final int checksumOrder = compareByChecksum(a, b);
|
|
if (checksumOrder != 0) return checksumOrder;
|
|
final int createdOrder = a.fileCreatedAt.compareTo(b.fileCreatedAt);
|
|
if (createdOrder != 0) return createdOrder;
|
|
return a.fileModifiedAt.compareTo(b.fileModifiedAt);
|
|
}
|
|
|
|
@override
|
|
String toString() {
|
|
return """
|
|
{
|
|
"id": ${id == Isar.autoIncrement ? '"N/A"' : id},
|
|
"remoteId": "${remoteId ?? "N/A"}",
|
|
"localId": "${localId ?? "N/A"}",
|
|
"checksum": "$checksum",
|
|
"ownerId": $ownerId,
|
|
"livePhotoVideoId": "${livePhotoVideoId ?? "N/A"}",
|
|
"stackId": "${stackId ?? "N/A"}",
|
|
"stackPrimaryAssetId": "${stackPrimaryAssetId ?? "N/A"}",
|
|
"stackCount": "$stackCount",
|
|
"fileCreatedAt": "$fileCreatedAt",
|
|
"fileModifiedAt": "$fileModifiedAt",
|
|
"updatedAt": "$updatedAt",
|
|
"durationInSeconds": $durationInSeconds,
|
|
"type": "$type",
|
|
"fileName": "$fileName",
|
|
"isFavorite": $isFavorite,
|
|
"isRemote": $isRemote,
|
|
"storage": "$storage",
|
|
"width": ${width ?? "N/A"},
|
|
"height": ${height ?? "N/A"},
|
|
"isArchived": $isArchived,
|
|
"isTrashed": $isTrashed,
|
|
"isOffline": $isOffline,
|
|
}""";
|
|
}
|
|
}
|
|
|
|
enum AssetType {
|
|
// do not change this order!
|
|
other,
|
|
image,
|
|
video,
|
|
audio,
|
|
}
|
|
|
|
extension AssetTypeEnumHelper on AssetTypeEnum {
|
|
AssetType toAssetType() {
|
|
switch (this) {
|
|
case AssetTypeEnum.IMAGE:
|
|
return AssetType.image;
|
|
case AssetTypeEnum.VIDEO:
|
|
return AssetType.video;
|
|
case AssetTypeEnum.AUDIO:
|
|
return AssetType.audio;
|
|
case AssetTypeEnum.OTHER:
|
|
return AssetType.other;
|
|
}
|
|
throw Exception();
|
|
}
|
|
}
|
|
|
|
/// Describes where the information of this asset came from:
|
|
/// only from the local device, only from the remote server or merged from both
|
|
enum AssetState {
|
|
local,
|
|
remote,
|
|
merged,
|
|
}
|
|
|
|
extension AssetsHelper on IsarCollection<Asset> {
|
|
Future<int> deleteAllByRemoteId(Iterable<String> ids) =>
|
|
ids.isEmpty ? Future.value(0) : remote(ids).deleteAll();
|
|
Future<int> deleteAllByLocalId(Iterable<String> ids) =>
|
|
ids.isEmpty ? Future.value(0) : local(ids).deleteAll();
|
|
Future<List<Asset>> getAllByRemoteId(Iterable<String> ids) =>
|
|
ids.isEmpty ? Future.value([]) : remote(ids).findAll();
|
|
Future<List<Asset>> getAllByLocalId(Iterable<String> ids) =>
|
|
ids.isEmpty ? Future.value([]) : local(ids).findAll();
|
|
Future<Asset?> getByRemoteId(String id) =>
|
|
where().remoteIdEqualTo(id).findFirst();
|
|
|
|
QueryBuilder<Asset, Asset, QAfterWhereClause> remote(
|
|
Iterable<String> ids,
|
|
) =>
|
|
where().anyOf(ids, (q, String e) => q.remoteIdEqualTo(e));
|
|
QueryBuilder<Asset, Asset, QAfterWhereClause> local(
|
|
Iterable<String> ids,
|
|
) {
|
|
return where().anyOf(ids, (q, String e) => q.localIdEqualTo(e));
|
|
}
|
|
}
|