mirror of
https://gitlab.com/nkming2/nc-photos.git
synced 2025-01-26 02:46:19 +01:00
253 lines
9 KiB
Dart
253 lines
9 KiB
Dart
import 'dart:convert';
|
|
|
|
import 'package:drift/drift.dart';
|
|
import 'package:nc_photos/ci_string.dart';
|
|
import 'package:nc_photos/entity/album.dart';
|
|
import 'package:nc_photos/entity/album/cover_provider.dart';
|
|
import 'package:nc_photos/entity/album/provider.dart';
|
|
import 'package:nc_photos/entity/album/sort_provider.dart';
|
|
import 'package:nc_photos/entity/exif.dart';
|
|
import 'package:nc_photos/entity/file.dart';
|
|
import 'package:nc_photos/entity/file_descriptor.dart';
|
|
import 'package:nc_photos/entity/person.dart';
|
|
import 'package:nc_photos/entity/sqlite_table.dart' as sql;
|
|
import 'package:nc_photos/entity/sqlite_table_extension.dart' as sql;
|
|
import 'package:nc_photos/entity/tag.dart';
|
|
import 'package:nc_photos/iterable_extension.dart';
|
|
import 'package:nc_photos/object_extension.dart';
|
|
import 'package:nc_photos/or_null.dart';
|
|
|
|
extension SqlTagListExtension on List<sql.Tag> {
|
|
Future<List<Tag>> convertToAppTag() {
|
|
return computeAll(SqliteTagConverter.fromSql);
|
|
}
|
|
}
|
|
|
|
extension AppTagListExtension on List<Tag> {
|
|
Future<List<sql.TagsCompanion>> convertToTagCompanion(
|
|
sql.Account? dbAccount) {
|
|
return map((t) => {
|
|
"account": dbAccount,
|
|
"tag": t,
|
|
}).computeAll(_convertAppTag);
|
|
}
|
|
}
|
|
|
|
extension SqlPersonListExtension on List<sql.Person> {
|
|
Future<List<Person>> convertToAppPerson() {
|
|
return computeAll(SqlitePersonConverter.fromSql);
|
|
}
|
|
}
|
|
|
|
extension AppPersonListExtension on List<Person> {
|
|
Future<List<sql.PersonsCompanion>> convertToPersonCompanion(
|
|
sql.Account? dbAccount) {
|
|
return map((p) => {
|
|
"account": dbAccount,
|
|
"person": p,
|
|
}).computeAll(_convertAppPerson);
|
|
}
|
|
}
|
|
|
|
class SqliteAlbumConverter {
|
|
static Album fromSql(
|
|
sql.Album album, File albumFile, List<sql.AlbumShare> shares) {
|
|
return Album(
|
|
lastUpdated: album.lastUpdated,
|
|
name: album.name,
|
|
provider: AlbumProvider.fromJson({
|
|
"type": album.providerType,
|
|
"content": jsonDecode(album.providerContent),
|
|
}),
|
|
coverProvider: AlbumCoverProvider.fromJson({
|
|
"type": album.coverProviderType,
|
|
"content": jsonDecode(album.coverProviderContent),
|
|
}),
|
|
sortProvider: AlbumSortProvider.fromJson({
|
|
"type": album.sortProviderType,
|
|
"content": jsonDecode(album.sortProviderContent),
|
|
}),
|
|
shares: shares.isEmpty
|
|
? null
|
|
: shares
|
|
.map((e) => AlbumShare(
|
|
userId: e.userId.toCi(),
|
|
displayName: e.displayName,
|
|
sharedAt: e.sharedAt.toUtc(),
|
|
))
|
|
.toList(),
|
|
// replace with the original etag when this album was cached
|
|
albumFile: albumFile.copyWith(etag: OrNull(album.fileEtag)),
|
|
savedVersion: album.version,
|
|
);
|
|
}
|
|
|
|
static sql.CompleteAlbumCompanion toSql(
|
|
Album album, int albumFileRowId, String albumFileEtag) {
|
|
final providerJson = album.provider.toJson();
|
|
final coverProviderJson = album.coverProvider.toJson();
|
|
final sortProviderJson = album.sortProvider.toJson();
|
|
final dbAlbum = sql.AlbumsCompanion.insert(
|
|
file: albumFileRowId,
|
|
fileEtag: Value(albumFileEtag),
|
|
version: Album.version,
|
|
lastUpdated: album.lastUpdated,
|
|
name: album.name,
|
|
providerType: providerJson["type"],
|
|
providerContent: jsonEncode(providerJson["content"]),
|
|
coverProviderType: coverProviderJson["type"],
|
|
coverProviderContent: jsonEncode(coverProviderJson["content"]),
|
|
sortProviderType: sortProviderJson["type"],
|
|
sortProviderContent: jsonEncode(sortProviderJson["content"]),
|
|
);
|
|
final dbAlbumShares = album.shares
|
|
?.map((s) => sql.AlbumSharesCompanion(
|
|
userId: Value(s.userId.toCaseInsensitiveString()),
|
|
displayName: Value(s.displayName),
|
|
sharedAt: Value(s.sharedAt),
|
|
))
|
|
.toList();
|
|
return sql.CompleteAlbumCompanion(dbAlbum, dbAlbumShares ?? []);
|
|
}
|
|
}
|
|
|
|
class SqliteFileConverter {
|
|
static File fromSql(String userId, sql.CompleteFile f) {
|
|
final metadata = f.image?.run((obj) => Metadata(
|
|
lastUpdated: obj.lastUpdated,
|
|
fileEtag: obj.fileEtag,
|
|
imageWidth: obj.width,
|
|
imageHeight: obj.height,
|
|
exif: obj.exifRaw?.run((e) => Exif.fromJson(jsonDecode(e))),
|
|
));
|
|
final location = f.imageLocation?.run((obj) => ImageLocation(
|
|
version: obj.version,
|
|
name: obj.name,
|
|
latitude: obj.latitude,
|
|
longitude: obj.longitude,
|
|
countryCode: obj.countryCode,
|
|
admin1: obj.admin1,
|
|
admin2: obj.admin2,
|
|
));
|
|
return File(
|
|
path: "remote.php/dav/files/$userId/${f.accountFile.relativePath}",
|
|
contentLength: f.file.contentLength,
|
|
contentType: f.file.contentType,
|
|
etag: f.file.etag,
|
|
lastModified: f.file.lastModified,
|
|
isCollection: f.file.isCollection,
|
|
usedBytes: f.file.usedBytes,
|
|
hasPreview: f.file.hasPreview,
|
|
fileId: f.file.fileId,
|
|
isFavorite: f.accountFile.isFavorite,
|
|
ownerId: f.file.ownerId?.toCi(),
|
|
ownerDisplayName: f.file.ownerDisplayName,
|
|
trashbinFilename: f.trash?.filename,
|
|
trashbinOriginalLocation: f.trash?.originalLocation,
|
|
trashbinDeletionTime: f.trash?.deletionTime,
|
|
metadata: metadata,
|
|
isArchived: f.accountFile.isArchived,
|
|
overrideDateTime: f.accountFile.overrideDateTime,
|
|
location: location,
|
|
);
|
|
}
|
|
|
|
static sql.CompleteFileCompanion toSql(sql.Account? account, File file) {
|
|
final dbFile = sql.FilesCompanion(
|
|
server: account == null ? const Value.absent() : Value(account.server),
|
|
fileId: Value(file.fileId!),
|
|
contentLength: Value(file.contentLength),
|
|
contentType: Value(file.contentType),
|
|
etag: Value(file.etag),
|
|
lastModified: Value(file.lastModified),
|
|
isCollection: Value(file.isCollection),
|
|
usedBytes: Value(file.usedBytes),
|
|
hasPreview: Value(file.hasPreview),
|
|
ownerId: Value(file.ownerId!.toCaseInsensitiveString()),
|
|
ownerDisplayName: Value(file.ownerDisplayName),
|
|
);
|
|
final dbAccountFile = sql.AccountFilesCompanion(
|
|
account: account == null ? const Value.absent() : Value(account.rowId),
|
|
relativePath: Value(file.strippedPathWithEmpty),
|
|
isFavorite: Value(file.isFavorite),
|
|
isArchived: Value(file.isArchived),
|
|
overrideDateTime: Value(file.overrideDateTime),
|
|
bestDateTime: Value(file.bestDateTime),
|
|
);
|
|
final dbImage = file.metadata?.run((m) => sql.ImagesCompanion.insert(
|
|
lastUpdated: m.lastUpdated,
|
|
fileEtag: Value(m.fileEtag),
|
|
width: Value(m.imageWidth),
|
|
height: Value(m.imageHeight),
|
|
exifRaw: Value(m.exif?.toJson().run((j) => jsonEncode(j))),
|
|
dateTimeOriginal: Value(m.exif?.dateTimeOriginal),
|
|
));
|
|
final dbImageLocation =
|
|
file.location?.run((l) => sql.ImageLocationsCompanion.insert(
|
|
version: l.version,
|
|
name: Value(l.name),
|
|
latitude: Value(l.latitude),
|
|
longitude: Value(l.longitude),
|
|
countryCode: Value(l.countryCode),
|
|
admin1: Value(l.admin1),
|
|
admin2: Value(l.admin2),
|
|
));
|
|
final dbTrash = file.trashbinDeletionTime == null
|
|
? null
|
|
: sql.TrashesCompanion.insert(
|
|
filename: file.trashbinFilename!,
|
|
originalLocation: file.trashbinOriginalLocation!,
|
|
deletionTime: file.trashbinDeletionTime!,
|
|
);
|
|
return sql.CompleteFileCompanion(
|
|
dbFile, dbAccountFile, dbImage, dbImageLocation, dbTrash);
|
|
}
|
|
}
|
|
|
|
class SqliteTagConverter {
|
|
static Tag fromSql(sql.Tag tag) => Tag(
|
|
id: tag.tagId,
|
|
displayName: tag.displayName,
|
|
userVisible: tag.userVisible,
|
|
userAssignable: tag.userAssignable,
|
|
);
|
|
|
|
static sql.TagsCompanion toSql(sql.Account? dbAccount, Tag tag) =>
|
|
sql.TagsCompanion(
|
|
server:
|
|
dbAccount == null ? const Value.absent() : Value(dbAccount.server),
|
|
tagId: Value(tag.id),
|
|
displayName: Value(tag.displayName),
|
|
userVisible: Value(tag.userVisible),
|
|
userAssignable: Value(tag.userAssignable),
|
|
);
|
|
}
|
|
|
|
class SqlitePersonConverter {
|
|
static Person fromSql(sql.Person person) => Person(
|
|
name: person.name,
|
|
thumbFaceId: person.thumbFaceId,
|
|
count: person.count,
|
|
);
|
|
|
|
static sql.PersonsCompanion toSql(sql.Account? dbAccount, Person person) =>
|
|
sql.PersonsCompanion(
|
|
account:
|
|
dbAccount == null ? const Value.absent() : Value(dbAccount.rowId),
|
|
name: Value(person.name),
|
|
thumbFaceId: Value(person.thumbFaceId),
|
|
count: Value(person.count),
|
|
);
|
|
}
|
|
|
|
sql.TagsCompanion _convertAppTag(Map map) {
|
|
final account = map["account"] as sql.Account?;
|
|
final tag = map["tag"] as Tag;
|
|
return SqliteTagConverter.toSql(account, tag);
|
|
}
|
|
|
|
sql.PersonsCompanion _convertAppPerson(Map map) {
|
|
final account = map["account"] as sql.Account?;
|
|
final person = map["person"] as Person;
|
|
return SqlitePersonConverter.toSql(account, person);
|
|
}
|