1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-19 18:26:46 +01:00

feat(mobile): backup to album

This commit is contained in:
Alex 2024-06-16 08:57:39 -07:00
parent 1cd51cc2de
commit a3e8701f0a
No known key found for this signature in database
GPG key ID: 53CD082B3A5E1082
3 changed files with 62 additions and 12 deletions

View file

@ -0,0 +1,36 @@
// ignore_for_file: public_member_api_docs, sort_constructors_first
import 'package:photo_manager/photo_manager.dart';
class BackupCandidate {
final String albumName;
final AssetEntity asset;
BackupCandidate({
required this.albumName,
required this.asset,
});
BackupCandidate copyWith({
String? albumName,
AssetEntity? asset,
}) {
return BackupCandidate(
albumName: albumName ?? this.albumName,
asset: asset ?? this.asset,
);
}
@override
String toString() => 'BackupCandidate(albumName: $albumName, asset: $asset)';
@override
bool operator ==(covariant BackupCandidate other) {
if (identical(this, other)) return true;
return other.albumName == albumName && other.asset == asset;
}
@override
int get hashCode => albumName.hashCode ^ asset.hashCode;
}

View file

@ -2,7 +2,7 @@
import 'package:cancellation_token_http/http.dart'; import 'package:cancellation_token_http/http.dart';
import 'package:collection/collection.dart'; import 'package:collection/collection.dart';
import 'package:photo_manager/photo_manager.dart'; import 'package:immich_mobile/models/backup/backup_candidate.model.dart';
import 'package:immich_mobile/models/backup/available_album.model.dart'; import 'package:immich_mobile/models/backup/available_album.model.dart';
import 'package:immich_mobile/models/backup/current_upload_asset.model.dart'; import 'package:immich_mobile/models/backup/current_upload_asset.model.dart';
@ -41,7 +41,7 @@ class BackUpState {
final Set<AvailableAlbum> excludedBackupAlbums; final Set<AvailableAlbum> excludedBackupAlbums;
/// Assets that are not overlapping in selected backup albums and excluded backup albums /// Assets that are not overlapping in selected backup albums and excluded backup albums
final Set<AssetEntity> allUniqueAssets; final Set<BackupCandidate> allUniqueAssets;
/// All assets from the selected albums that have been backup /// All assets from the selected albums that have been backup
final Set<String> selectedAlbumsBackupAssetsIds; final Set<String> selectedAlbumsBackupAssetsIds;
@ -94,7 +94,7 @@ class BackUpState {
List<AvailableAlbum>? availableAlbums, List<AvailableAlbum>? availableAlbums,
Set<AvailableAlbum>? selectedBackupAlbums, Set<AvailableAlbum>? selectedBackupAlbums,
Set<AvailableAlbum>? excludedBackupAlbums, Set<AvailableAlbum>? excludedBackupAlbums,
Set<AssetEntity>? allUniqueAssets, Set<BackupCandidate>? allUniqueAssets,
Set<String>? selectedAlbumsBackupAssetsIds, Set<String>? selectedAlbumsBackupAssetsIds,
CurrentUploadAsset? currentUploadAsset, CurrentUploadAsset? currentUploadAsset,
}) { }) {

View file

@ -6,6 +6,7 @@ import 'package:flutter/widgets.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/models/backup/available_album.model.dart'; import 'package:immich_mobile/models/backup/available_album.model.dart';
import 'package:immich_mobile/entities/backup_album.entity.dart'; import 'package:immich_mobile/entities/backup_album.entity.dart';
import 'package:immich_mobile/models/backup/backup_candidate.model.dart';
import 'package:immich_mobile/models/backup/backup_state.model.dart'; import 'package:immich_mobile/models/backup/backup_state.model.dart';
import 'package:immich_mobile/models/backup/current_upload_asset.model.dart'; import 'package:immich_mobile/models/backup/current_upload_asset.model.dart';
import 'package:immich_mobile/models/backup/error_upload_asset.model.dart'; import 'package:immich_mobile/models/backup/error_upload_asset.model.dart';
@ -289,9 +290,10 @@ class BackupNotifier extends StateNotifier<BackUpState> {
/// Those assets are unique and are used as the total assets /// Those assets are unique and are used as the total assets
/// ///
Future<void> _updateBackupAssetCount() async { Future<void> _updateBackupAssetCount() async {
debugPrint("UPDATE BACKUP ASSET COUNTTT");
final duplicatedAssetIds = await _backupService.getDuplicatedAssetIds(); final duplicatedAssetIds = await _backupService.getDuplicatedAssetIds();
final Set<AssetEntity> assetsFromSelectedAlbums = {}; final Set<BackupCandidate> assetsFromSelectedAlbums = {};
final Set<AssetEntity> assetsFromExcludedAlbums = {}; final Set<BackupCandidate> assetsFromExcludedAlbums = {};
for (final album in state.selectedBackupAlbums) { for (final album in state.selectedBackupAlbums) {
final assetCount = await album.albumEntity.assetCountAsync; final assetCount = await album.albumEntity.assetCountAsync;
@ -304,7 +306,12 @@ class BackupNotifier extends StateNotifier<BackUpState> {
start: 0, start: 0,
end: assetCount, end: assetCount,
); );
assetsFromSelectedAlbums.addAll(assets);
final candidate = assets.map(
(e) => BackupCandidate(albumName: album.albumEntity.name, asset: e),
);
assetsFromSelectedAlbums.addAll(candidate.toSet());
} }
for (final album in state.excludedBackupAlbums) { for (final album in state.excludedBackupAlbums) {
@ -318,10 +325,15 @@ class BackupNotifier extends StateNotifier<BackUpState> {
start: 0, start: 0,
end: assetCount, end: assetCount,
); );
assetsFromExcludedAlbums.addAll(assets);
final candidate = assets.map(
(e) => BackupCandidate(albumName: album.albumEntity.name, asset: e),
);
assetsFromExcludedAlbums.addAll(candidate);
} }
final Set<AssetEntity> allUniqueAssets = final Set<BackupCandidate> allUniqueAssets =
assetsFromSelectedAlbums.difference(assetsFromExcludedAlbums); assetsFromSelectedAlbums.difference(assetsFromExcludedAlbums);
final allAssetsInDatabase = await _backupService.getDeviceBackupAsset(); final allAssetsInDatabase = await _backupService.getDeviceBackupAsset();
@ -331,14 +343,14 @@ class BackupNotifier extends StateNotifier<BackUpState> {
// Find asset that were backup from selected albums // Find asset that were backup from selected albums
final Set<String> selectedAlbumsBackupAssets = final Set<String> selectedAlbumsBackupAssets =
Set.from(allUniqueAssets.map((e) => e.id)); Set.from(allUniqueAssets.map((e) => e.asset.id));
selectedAlbumsBackupAssets selectedAlbumsBackupAssets
.removeWhere((assetId) => !allAssetsInDatabase.contains(assetId)); .removeWhere((assetId) => !allAssetsInDatabase.contains(assetId));
// Remove duplicated asset from all unique assets // Remove duplicated asset from all unique assets
allUniqueAssets.removeWhere( allUniqueAssets.removeWhere(
(asset) => duplicatedAssetIds.contains(asset.id), (e) => duplicatedAssetIds.contains(e.asset.id),
); );
if (allUniqueAssets.isEmpty) { if (allUniqueAssets.isEmpty) {
@ -359,6 +371,8 @@ class BackupNotifier extends StateNotifier<BackUpState> {
// Save to persistent storage // Save to persistent storage
await _updatePersistentAlbumsSelection(); await _updatePersistentAlbumsSelection();
debugPrint("backup asset $allUniqueAssets", wrapWidth: 80);
} }
/// Get all necessary information for calculating the available albums, /// Get all necessary information for calculating the available albums,
@ -505,7 +519,7 @@ class BackupNotifier extends StateNotifier<BackUpState> {
if (isDuplicated) { if (isDuplicated) {
state = state.copyWith( state = state.copyWith(
allUniqueAssets: state.allUniqueAssets allUniqueAssets: state.allUniqueAssets
.where((asset) => asset.id != deviceAssetId) .where((e) => e.asset.id != deviceAssetId)
.toSet(), .toSet(),
); );
} else { } else {
@ -522,7 +536,7 @@ class BackupNotifier extends StateNotifier<BackUpState> {
state.selectedAlbumsBackupAssetsIds.length == state.selectedAlbumsBackupAssetsIds.length ==
0) { 0) {
final latestAssetBackup = final latestAssetBackup =
state.allUniqueAssets.map((e) => e.modifiedDateTime).reduce( state.allUniqueAssets.map((e) => e.asset.modifiedDateTime).reduce(
(v, e) => e.isAfter(v) ? e : v, (v, e) => e.isAfter(v) ? e : v,
); );
state = state.copyWith( state = state.copyWith(