2023-12-07 16:38:22 +01:00
|
|
|
import 'package:flutter/foundation.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
|
|
|
import 'package:immich_mobile/entities/user.entity.dart';
|
2024-04-02 07:22:15 +02:00
|
|
|
import 'package:immich_mobile/utils/datetime_comparison.dart';
|
2023-03-03 23:38:30 +01:00
|
|
|
import 'package:isar/isar.dart';
|
2024-09-20 15:32:37 +02:00
|
|
|
// ignore: implementation_imports
|
|
|
|
import 'package:isar/src/common/isar_links_common.dart';
|
2023-02-06 08:13:32 +01:00
|
|
|
import 'package:openapi/api.dart';
|
|
|
|
|
2024-05-01 04:36:40 +02:00
|
|
|
part 'album.entity.g.dart';
|
2023-02-06 08:13:32 +01:00
|
|
|
|
2023-03-03 23:38:30 +01:00
|
|
|
@Collection(inheritance: false)
|
|
|
|
class Album {
|
|
|
|
@protected
|
2023-02-06 08:13:32 +01:00
|
|
|
Album({
|
|
|
|
this.remoteId,
|
|
|
|
this.localId,
|
|
|
|
required this.name,
|
|
|
|
required this.createdAt,
|
|
|
|
required this.modifiedAt,
|
2023-10-25 23:56:31 +02:00
|
|
|
this.startDate,
|
|
|
|
this.endDate,
|
2023-10-12 20:18:54 +02:00
|
|
|
this.lastModifiedAssetTimestamp,
|
2023-02-06 08:13:32 +01:00
|
|
|
required this.shared,
|
2023-11-08 04:07:43 +01:00
|
|
|
required this.activityEnabled,
|
2023-02-06 08:13:32 +01:00
|
|
|
});
|
|
|
|
|
2024-09-20 15:32:37 +02:00
|
|
|
// fields stored in DB
|
2023-03-03 23:38:30 +01:00
|
|
|
Id id = Isar.autoIncrement;
|
|
|
|
@Index(unique: false, replace: false, type: IndexType.hash)
|
2023-02-06 08:13:32 +01:00
|
|
|
String? remoteId;
|
2023-03-03 23:38:30 +01:00
|
|
|
@Index(unique: false, replace: false, type: IndexType.hash)
|
2023-02-06 08:13:32 +01:00
|
|
|
String? localId;
|
|
|
|
String name;
|
|
|
|
DateTime createdAt;
|
|
|
|
DateTime modifiedAt;
|
2023-10-25 23:56:31 +02:00
|
|
|
DateTime? startDate;
|
|
|
|
DateTime? endDate;
|
2023-10-12 20:18:54 +02:00
|
|
|
DateTime? lastModifiedAssetTimestamp;
|
2023-02-06 08:13:32 +01:00
|
|
|
bool shared;
|
2023-11-08 04:07:43 +01:00
|
|
|
bool activityEnabled;
|
2023-03-03 23:38:30 +01:00
|
|
|
final IsarLink<User> owner = IsarLink<User>();
|
|
|
|
final IsarLink<Asset> thumbnail = IsarLink<Asset>();
|
|
|
|
final IsarLinks<User> sharedUsers = IsarLinks<User>();
|
|
|
|
final IsarLinks<Asset> assets = IsarLinks<Asset>();
|
|
|
|
|
2024-09-20 15:32:37 +02:00
|
|
|
// transient fields
|
2024-09-18 17:15:52 +02:00
|
|
|
@ignore
|
|
|
|
bool isAll = false;
|
|
|
|
|
2024-09-20 15:32:37 +02:00
|
|
|
@ignore
|
|
|
|
String? remoteThumbnailAssetId;
|
|
|
|
|
|
|
|
@ignore
|
|
|
|
int remoteAssetCount = 0;
|
|
|
|
|
|
|
|
// getters
|
2023-03-03 23:38:30 +01:00
|
|
|
@ignore
|
2023-02-06 08:13:32 +01:00
|
|
|
bool get isRemote => remoteId != null;
|
|
|
|
|
2023-03-03 23:38:30 +01:00
|
|
|
@ignore
|
2023-02-06 08:13:32 +01:00
|
|
|
bool get isLocal => localId != null;
|
|
|
|
|
2023-03-03 23:38:30 +01:00
|
|
|
@ignore
|
|
|
|
int get assetCount => assets.length;
|
|
|
|
|
|
|
|
@ignore
|
|
|
|
String? get ownerId => owner.value?.id;
|
|
|
|
|
2023-03-19 20:47:51 +01:00
|
|
|
@ignore
|
|
|
|
String? get ownerName {
|
|
|
|
// Guard null owner
|
|
|
|
if (owner.value == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
final name = <String>[];
|
2023-11-12 02:03:32 +01:00
|
|
|
if (owner.value?.name != null) {
|
|
|
|
name.add(owner.value!.name);
|
2023-03-19 20:47:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return name.join(' ');
|
|
|
|
}
|
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
@ignore
|
|
|
|
String get eTagKeyAssetCount => "device-album-$localId-asset-count";
|
|
|
|
|
2024-09-20 15:32:37 +02:00
|
|
|
// the following getter are needed because Isar links do not make data
|
|
|
|
// accessible in an object freshly created (not loaded from DB)
|
|
|
|
|
|
|
|
@ignore
|
|
|
|
Iterable<User> get remoteUsers => sharedUsers.isEmpty
|
|
|
|
? (sharedUsers as IsarLinksCommon<User>).addedObjects
|
|
|
|
: sharedUsers;
|
|
|
|
|
|
|
|
@ignore
|
|
|
|
Iterable<Asset> get remoteAssets =>
|
|
|
|
assets.isEmpty ? (assets as IsarLinksCommon<Asset>).addedObjects : assets;
|
|
|
|
|
2023-02-06 08:13:32 +01:00
|
|
|
@override
|
|
|
|
bool operator ==(other) {
|
|
|
|
if (other is! Album) return false;
|
2023-03-03 23:38:30 +01:00
|
|
|
return id == other.id &&
|
|
|
|
remoteId == other.remoteId &&
|
2023-02-06 08:13:32 +01:00
|
|
|
localId == other.localId &&
|
|
|
|
name == other.name &&
|
2023-05-25 05:52:43 +02:00
|
|
|
createdAt.isAtSameMomentAs(other.createdAt) &&
|
|
|
|
modifiedAt.isAtSameMomentAs(other.modifiedAt) &&
|
2024-04-02 07:22:15 +02:00
|
|
|
isAtSameMomentAs(startDate, other.startDate) &&
|
|
|
|
isAtSameMomentAs(endDate, other.endDate) &&
|
|
|
|
isAtSameMomentAs(
|
|
|
|
lastModifiedAssetTimestamp,
|
|
|
|
other.lastModifiedAssetTimestamp,
|
|
|
|
) &&
|
2023-02-06 08:13:32 +01:00
|
|
|
shared == other.shared &&
|
2023-11-08 04:07:43 +01:00
|
|
|
activityEnabled == other.activityEnabled &&
|
2023-03-03 23:38:30 +01:00
|
|
|
owner.value == other.owner.value &&
|
|
|
|
thumbnail.value == other.thumbnail.value &&
|
|
|
|
sharedUsers.length == other.sharedUsers.length &&
|
|
|
|
assets.length == other.assets.length;
|
2023-02-06 08:13:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2023-03-03 23:38:30 +01:00
|
|
|
@ignore
|
2023-02-06 08:13:32 +01:00
|
|
|
int get hashCode =>
|
2023-03-03 23:38:30 +01:00
|
|
|
id.hashCode ^
|
2023-02-06 08:13:32 +01:00
|
|
|
remoteId.hashCode ^
|
|
|
|
localId.hashCode ^
|
|
|
|
name.hashCode ^
|
|
|
|
createdAt.hashCode ^
|
|
|
|
modifiedAt.hashCode ^
|
2024-04-02 07:22:15 +02:00
|
|
|
startDate.hashCode ^
|
|
|
|
endDate.hashCode ^
|
2023-10-12 20:18:54 +02:00
|
|
|
lastModifiedAssetTimestamp.hashCode ^
|
2023-02-06 08:13:32 +01:00
|
|
|
shared.hashCode ^
|
2023-11-08 04:07:43 +01:00
|
|
|
activityEnabled.hashCode ^
|
2023-03-03 23:38:30 +01:00
|
|
|
owner.value.hashCode ^
|
|
|
|
thumbnail.value.hashCode ^
|
|
|
|
sharedUsers.length.hashCode ^
|
|
|
|
assets.length.hashCode;
|
|
|
|
|
|
|
|
static Future<Album> remote(AlbumResponseDto dto) async {
|
|
|
|
final Isar db = Isar.getInstance()!;
|
|
|
|
final Album a = Album(
|
|
|
|
remoteId: dto.id,
|
|
|
|
name: dto.albumName,
|
2023-05-30 15:15:56 +02:00
|
|
|
createdAt: dto.createdAt,
|
|
|
|
modifiedAt: dto.updatedAt,
|
2023-10-12 20:18:54 +02:00
|
|
|
lastModifiedAssetTimestamp: dto.lastModifiedAssetTimestamp,
|
2023-03-03 23:38:30 +01:00
|
|
|
shared: dto.shared,
|
2023-10-25 23:56:31 +02:00
|
|
|
startDate: dto.startDate,
|
|
|
|
endDate: dto.endDate,
|
2023-11-08 04:07:43 +01:00
|
|
|
activityEnabled: dto.isActivityEnabled,
|
2023-03-03 23:38:30 +01:00
|
|
|
);
|
2024-09-20 15:32:37 +02:00
|
|
|
a.remoteAssetCount = dto.assetCount;
|
2023-03-03 23:38:30 +01:00
|
|
|
a.owner.value = await db.users.getById(dto.ownerId);
|
|
|
|
if (dto.albumThumbnailAssetId != null) {
|
|
|
|
a.thumbnail.value = await db.assets
|
|
|
|
.where()
|
|
|
|
.remoteIdEqualTo(dto.albumThumbnailAssetId)
|
|
|
|
.findFirst();
|
2023-02-06 08:13:32 +01:00
|
|
|
}
|
2024-05-24 16:37:01 +02:00
|
|
|
if (dto.albumUsers.isNotEmpty) {
|
|
|
|
final users = await db.users.getAllById(
|
|
|
|
dto.albumUsers.map((e) => e.user.id).toList(growable: false),
|
|
|
|
);
|
2023-03-03 23:38:30 +01:00
|
|
|
a.sharedUsers.addAll(users.cast());
|
|
|
|
}
|
|
|
|
if (dto.assets.isNotEmpty) {
|
|
|
|
final assets =
|
|
|
|
await db.assets.getAllByRemoteId(dto.assets.map((e) => e.id));
|
|
|
|
a.assets.addAll(assets);
|
|
|
|
}
|
|
|
|
return a;
|
2023-02-06 08:13:32 +01:00
|
|
|
}
|
2023-03-19 23:05:18 +01:00
|
|
|
|
|
|
|
@override
|
|
|
|
String toString() => name;
|
2023-02-06 08:13:32 +01:00
|
|
|
}
|
|
|
|
|
2023-03-03 23:38:30 +01:00
|
|
|
extension AssetsHelper on IsarCollection<Album> {
|
2024-09-16 22:26:14 +02:00
|
|
|
Future<Album> store(Album a) async {
|
2023-03-03 23:38:30 +01:00
|
|
|
await put(a);
|
|
|
|
await a.owner.save();
|
|
|
|
await a.thumbnail.save();
|
|
|
|
await a.sharedUsers.save();
|
|
|
|
await a.assets.save();
|
2024-09-16 22:26:14 +02:00
|
|
|
return a;
|
2023-03-03 23:38:30 +01:00
|
|
|
}
|
|
|
|
}
|