forked from Cutlery/immich
		
	feat(mobile): handle backup iCloud asset (#5508)
* feat(mobile): handle backup iCloud asset * additional state * Download progress * Added a separate page for backup options * handle ingore iCloud asset upload' * fix init backup service * PR feedback * fix negative count * get file title
This commit is contained in:
		
							parent
							
								
									c25556bb08
								
							
						
					
					
						commit
						2e59b07cc6
					
				| @ -169,4 +169,4 @@ SPEC CHECKSUMS: | |||||||
| 
 | 
 | ||||||
| PODFILE CHECKSUM: 599d8aeb73728400c15364e734525722250a5382 | PODFILE CHECKSUM: 599d8aeb73728400c15364e734525722250a5382 | ||||||
| 
 | 
 | ||||||
| COCOAPODS: 1.11.3 | COCOAPODS: 1.12.1 | ||||||
|  | |||||||
| @ -342,7 +342,8 @@ class BackgroundService { | |||||||
| 
 | 
 | ||||||
|     ApiService apiService = ApiService(); |     ApiService apiService = ApiService(); | ||||||
|     apiService.setAccessToken(Store.get(StoreKey.accessToken)); |     apiService.setAccessToken(Store.get(StoreKey.accessToken)); | ||||||
|     BackupService backupService = BackupService(apiService, db); |     AppSettingsService settingService = AppSettingsService(); | ||||||
|  |     BackupService backupService = BackupService(apiService, db, settingService); | ||||||
|     AppSettingsService settingsService = AppSettingsService(); |     AppSettingsService settingsService = AppSettingsService(); | ||||||
| 
 | 
 | ||||||
|     final selectedAlbums = backupService.selectedAlbumsQuery().findAllSync(); |     final selectedAlbums = backupService.selectedAlbumsQuery().findAllSync(); | ||||||
| @ -452,9 +453,12 @@ class BackgroundService { | |||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     _cancellationToken = CancellationToken(); |     _cancellationToken = CancellationToken(); | ||||||
|  |     final pmProgressHandler = PMProgressHandler(); | ||||||
|  | 
 | ||||||
|     final bool ok = await backupService.backupAsset( |     final bool ok = await backupService.backupAsset( | ||||||
|       toUpload, |       toUpload, | ||||||
|       _cancellationToken!, |       _cancellationToken!, | ||||||
|  |       pmProgressHandler, | ||||||
|       notifyTotalProgress ? _onAssetUploaded : (assetId, deviceId, isDup) {}, |       notifyTotalProgress ? _onAssetUploaded : (assetId, deviceId, isDup) {}, | ||||||
|       notifySingleProgress ? _onProgress : (sent, total) {}, |       notifySingleProgress ? _onProgress : (sent, total) {}, | ||||||
|       notifySingleProgress ? _onSetCurrentBackupAsset : (asset) {}, |       notifySingleProgress ? _onSetCurrentBackupAsset : (asset) {}, | ||||||
|  | |||||||
| @ -1,10 +1,12 @@ | |||||||
|  | // ignore_for_file: public_member_api_docs, sort_constructors_first | ||||||
|  | 
 | ||||||
| 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:immich_mobile/shared/models/server_info/server_disk_info.model.dart'; |  | ||||||
| import 'package:photo_manager/photo_manager.dart'; | import 'package:photo_manager/photo_manager.dart'; | ||||||
| 
 | 
 | ||||||
| import 'package:immich_mobile/modules/backup/models/available_album.model.dart'; | import 'package:immich_mobile/modules/backup/models/available_album.model.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/models/current_upload_asset.model.dart'; | import 'package:immich_mobile/modules/backup/models/current_upload_asset.model.dart'; | ||||||
|  | import 'package:immich_mobile/shared/models/server_info/server_disk_info.model.dart'; | ||||||
| 
 | 
 | ||||||
| enum BackUpProgressEnum { | enum BackUpProgressEnum { | ||||||
|   idle, |   idle, | ||||||
| @ -19,6 +21,7 @@ class BackUpState { | |||||||
|   final BackUpProgressEnum backupProgress; |   final BackUpProgressEnum backupProgress; | ||||||
|   final List<String> allAssetsInDatabase; |   final List<String> allAssetsInDatabase; | ||||||
|   final double progressInPercentage; |   final double progressInPercentage; | ||||||
|  |   final double iCloudDownloadProgress; | ||||||
|   final CancellationToken cancelToken; |   final CancellationToken cancelToken; | ||||||
|   final ServerDiskInfo serverInfo; |   final ServerDiskInfo serverInfo; | ||||||
|   final bool autoBackup; |   final bool autoBackup; | ||||||
| @ -45,6 +48,7 @@ class BackUpState { | |||||||
|     required this.backupProgress, |     required this.backupProgress, | ||||||
|     required this.allAssetsInDatabase, |     required this.allAssetsInDatabase, | ||||||
|     required this.progressInPercentage, |     required this.progressInPercentage, | ||||||
|  |     required this.iCloudDownloadProgress, | ||||||
|     required this.cancelToken, |     required this.cancelToken, | ||||||
|     required this.serverInfo, |     required this.serverInfo, | ||||||
|     required this.autoBackup, |     required this.autoBackup, | ||||||
| @ -64,6 +68,7 @@ class BackUpState { | |||||||
|     BackUpProgressEnum? backupProgress, |     BackUpProgressEnum? backupProgress, | ||||||
|     List<String>? allAssetsInDatabase, |     List<String>? allAssetsInDatabase, | ||||||
|     double? progressInPercentage, |     double? progressInPercentage, | ||||||
|  |     double? iCloudDownloadProgress, | ||||||
|     CancellationToken? cancelToken, |     CancellationToken? cancelToken, | ||||||
|     ServerDiskInfo? serverInfo, |     ServerDiskInfo? serverInfo, | ||||||
|     bool? autoBackup, |     bool? autoBackup, | ||||||
| @ -82,6 +87,8 @@ class BackUpState { | |||||||
|       backupProgress: backupProgress ?? this.backupProgress, |       backupProgress: backupProgress ?? this.backupProgress, | ||||||
|       allAssetsInDatabase: allAssetsInDatabase ?? this.allAssetsInDatabase, |       allAssetsInDatabase: allAssetsInDatabase ?? this.allAssetsInDatabase, | ||||||
|       progressInPercentage: progressInPercentage ?? this.progressInPercentage, |       progressInPercentage: progressInPercentage ?? this.progressInPercentage, | ||||||
|  |       iCloudDownloadProgress: | ||||||
|  |           iCloudDownloadProgress ?? this.iCloudDownloadProgress, | ||||||
|       cancelToken: cancelToken ?? this.cancelToken, |       cancelToken: cancelToken ?? this.cancelToken, | ||||||
|       serverInfo: serverInfo ?? this.serverInfo, |       serverInfo: serverInfo ?? this.serverInfo, | ||||||
|       autoBackup: autoBackup ?? this.autoBackup, |       autoBackup: autoBackup ?? this.autoBackup, | ||||||
| @ -102,18 +109,18 @@ class BackUpState { | |||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   String toString() { |   String toString() { | ||||||
|     return 'BackUpState(backupProgress: $backupProgress, allAssetsInDatabase: $allAssetsInDatabase, progressInPercentage: $progressInPercentage, cancelToken: $cancelToken, serverInfo: $serverInfo, autoBackup: $autoBackup, backgroundBackup: $backgroundBackup, backupRequireWifi: $backupRequireWifi, backupRequireCharging: $backupRequireCharging, backupTriggerDelay: $backupTriggerDelay, availableAlbums: $availableAlbums, selectedBackupAlbums: $selectedBackupAlbums, excludedBackupAlbums: $excludedBackupAlbums, allUniqueAssets: $allUniqueAssets, selectedAlbumsBackupAssetsIds: $selectedAlbumsBackupAssetsIds, currentUploadAsset: $currentUploadAsset)'; |     return 'BackUpState(backupProgress: $backupProgress, allAssetsInDatabase: $allAssetsInDatabase, progressInPercentage: $progressInPercentage, iCloudDownloadProgress: $iCloudDownloadProgress, cancelToken: $cancelToken, serverInfo: $serverInfo, autoBackup: $autoBackup, backgroundBackup: $backgroundBackup, backupRequireWifi: $backupRequireWifi, backupRequireCharging: $backupRequireCharging, backupTriggerDelay: $backupTriggerDelay, availableAlbums: $availableAlbums, selectedBackupAlbums: $selectedBackupAlbums, excludedBackupAlbums: $excludedBackupAlbums, allUniqueAssets: $allUniqueAssets, selectedAlbumsBackupAssetsIds: $selectedAlbumsBackupAssetsIds, currentUploadAsset: $currentUploadAsset)'; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   bool operator ==(Object other) { |   bool operator ==(covariant BackUpState other) { | ||||||
|     if (identical(this, other)) return true; |     if (identical(this, other)) return true; | ||||||
|     final collectionEquals = const DeepCollectionEquality().equals; |     final collectionEquals = const DeepCollectionEquality().equals; | ||||||
| 
 | 
 | ||||||
|     return other is BackUpState && |     return other.backupProgress == backupProgress && | ||||||
|         other.backupProgress == backupProgress && |  | ||||||
|         collectionEquals(other.allAssetsInDatabase, allAssetsInDatabase) && |         collectionEquals(other.allAssetsInDatabase, allAssetsInDatabase) && | ||||||
|         other.progressInPercentage == progressInPercentage && |         other.progressInPercentage == progressInPercentage && | ||||||
|  |         other.iCloudDownloadProgress == iCloudDownloadProgress && | ||||||
|         other.cancelToken == cancelToken && |         other.cancelToken == cancelToken && | ||||||
|         other.serverInfo == serverInfo && |         other.serverInfo == serverInfo && | ||||||
|         other.autoBackup == autoBackup && |         other.autoBackup == autoBackup && | ||||||
| @ -137,6 +144,7 @@ class BackUpState { | |||||||
|     return backupProgress.hashCode ^ |     return backupProgress.hashCode ^ | ||||||
|         allAssetsInDatabase.hashCode ^ |         allAssetsInDatabase.hashCode ^ | ||||||
|         progressInPercentage.hashCode ^ |         progressInPercentage.hashCode ^ | ||||||
|  |         iCloudDownloadProgress.hashCode ^ | ||||||
|         cancelToken.hashCode ^ |         cancelToken.hashCode ^ | ||||||
|         serverInfo.hashCode ^ |         serverInfo.hashCode ^ | ||||||
|         autoBackup.hashCode ^ |         autoBackup.hashCode ^ | ||||||
|  | |||||||
| @ -1,3 +1,4 @@ | |||||||
|  | // ignore_for_file: public_member_api_docs, sort_constructors_first | ||||||
| import 'dart:convert'; | import 'dart:convert'; | ||||||
| 
 | 
 | ||||||
| class CurrentUploadAsset { | class CurrentUploadAsset { | ||||||
| @ -5,12 +6,14 @@ class CurrentUploadAsset { | |||||||
|   final DateTime fileCreatedAt; |   final DateTime fileCreatedAt; | ||||||
|   final String fileName; |   final String fileName; | ||||||
|   final String fileType; |   final String fileType; | ||||||
|  |   final bool? iCloudAsset; | ||||||
| 
 | 
 | ||||||
|   CurrentUploadAsset({ |   CurrentUploadAsset({ | ||||||
|     required this.id, |     required this.id, | ||||||
|     required this.fileCreatedAt, |     required this.fileCreatedAt, | ||||||
|     required this.fileName, |     required this.fileName, | ||||||
|     required this.fileType, |     required this.fileType, | ||||||
|  |     this.iCloudAsset, | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   CurrentUploadAsset copyWith({ |   CurrentUploadAsset copyWith({ | ||||||
| @ -18,54 +21,58 @@ class CurrentUploadAsset { | |||||||
|     DateTime? fileCreatedAt, |     DateTime? fileCreatedAt, | ||||||
|     String? fileName, |     String? fileName, | ||||||
|     String? fileType, |     String? fileType, | ||||||
|  |     bool? iCloudAsset, | ||||||
|   }) { |   }) { | ||||||
|     return CurrentUploadAsset( |     return CurrentUploadAsset( | ||||||
|       id: id ?? this.id, |       id: id ?? this.id, | ||||||
|       fileCreatedAt: fileCreatedAt ?? this.fileCreatedAt, |       fileCreatedAt: fileCreatedAt ?? this.fileCreatedAt, | ||||||
|       fileName: fileName ?? this.fileName, |       fileName: fileName ?? this.fileName, | ||||||
|       fileType: fileType ?? this.fileType, |       fileType: fileType ?? this.fileType, | ||||||
|  |       iCloudAsset: iCloudAsset ?? this.iCloudAsset, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   Map<String, dynamic> toMap() { |   Map<String, dynamic> toMap() { | ||||||
|     final result = <String, dynamic>{}; |     return <String, dynamic>{ | ||||||
| 
 |       'id': id, | ||||||
|     result.addAll({'id': id}); |       'fileCreatedAt': fileCreatedAt.millisecondsSinceEpoch, | ||||||
|     result.addAll({'fileCreatedAt': fileCreatedAt.millisecondsSinceEpoch}); |       'fileName': fileName, | ||||||
|     result.addAll({'fileName': fileName}); |       'fileType': fileType, | ||||||
|     result.addAll({'fileType': fileType}); |       'iCloudAsset': iCloudAsset, | ||||||
| 
 |     }; | ||||||
|     return result; |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   factory CurrentUploadAsset.fromMap(Map<String, dynamic> map) { |   factory CurrentUploadAsset.fromMap(Map<String, dynamic> map) { | ||||||
|     return CurrentUploadAsset( |     return CurrentUploadAsset( | ||||||
|       id: map['id'] ?? '', |       id: map['id'] as String, | ||||||
|       fileCreatedAt: DateTime.fromMillisecondsSinceEpoch(map['fileCreatedAt']), |       fileCreatedAt: | ||||||
|       fileName: map['fileName'] ?? '', |           DateTime.fromMillisecondsSinceEpoch(map['fileCreatedAt'] as int), | ||||||
|       fileType: map['fileType'] ?? '', |       fileName: map['fileName'] as String, | ||||||
|  |       fileType: map['fileType'] as String, | ||||||
|  |       iCloudAsset: | ||||||
|  |           map['iCloudAsset'] != null ? map['iCloudAsset'] as bool : null, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   String toJson() => json.encode(toMap()); |   String toJson() => json.encode(toMap()); | ||||||
| 
 | 
 | ||||||
|   factory CurrentUploadAsset.fromJson(String source) => |   factory CurrentUploadAsset.fromJson(String source) => | ||||||
|       CurrentUploadAsset.fromMap(json.decode(source)); |       CurrentUploadAsset.fromMap(json.decode(source) as Map<String, dynamic>); | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   String toString() { |   String toString() { | ||||||
|     return 'CurrentUploadAsset(id: $id, fileCreatedAt: $fileCreatedAt, fileName: $fileName, fileType: $fileType)'; |     return 'CurrentUploadAsset(id: $id, fileCreatedAt: $fileCreatedAt, fileName: $fileName, fileType: $fileType, iCloudAsset: $iCloudAsset)'; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   bool operator ==(Object other) { |   bool operator ==(covariant CurrentUploadAsset other) { | ||||||
|     if (identical(this, other)) return true; |     if (identical(this, other)) return true; | ||||||
| 
 | 
 | ||||||
|     return other is CurrentUploadAsset && |     return other.id == id && | ||||||
|         other.id == id && |  | ||||||
|         other.fileCreatedAt == fileCreatedAt && |         other.fileCreatedAt == fileCreatedAt && | ||||||
|         other.fileName == fileName && |         other.fileName == fileName && | ||||||
|         other.fileType == fileType; |         other.fileType == fileType && | ||||||
|  |         other.iCloudAsset == iCloudAsset; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
| @ -73,6 +80,7 @@ class CurrentUploadAsset { | |||||||
|     return id.hashCode ^ |     return id.hashCode ^ | ||||||
|         fileCreatedAt.hashCode ^ |         fileCreatedAt.hashCode ^ | ||||||
|         fileName.hashCode ^ |         fileName.hashCode ^ | ||||||
|         fileType.hashCode; |         fileType.hashCode ^ | ||||||
|  |         iCloudAsset.hashCode; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -61,7 +61,9 @@ class BackupNotifier extends StateNotifier<BackUpState> { | |||||||
|               fileCreatedAt: DateTime.parse('2020-10-04'), |               fileCreatedAt: DateTime.parse('2020-10-04'), | ||||||
|               fileName: '...', |               fileName: '...', | ||||||
|               fileType: '...', |               fileType: '...', | ||||||
|  |               iCloudAsset: false, | ||||||
|             ), |             ), | ||||||
|  |             iCloudDownloadProgress: 0.0, | ||||||
|           ), |           ), | ||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
| @ -444,9 +446,18 @@ class BackupNotifier extends StateNotifier<BackUpState> { | |||||||
| 
 | 
 | ||||||
|       // Perform Backup |       // Perform Backup | ||||||
|       state = state.copyWith(cancelToken: CancellationToken()); |       state = state.copyWith(cancelToken: CancellationToken()); | ||||||
|  | 
 | ||||||
|  |       final pmProgressHandler = PMProgressHandler(); | ||||||
|  | 
 | ||||||
|  |       pmProgressHandler.stream.listen((event) { | ||||||
|  |         final double progress = event.progress; | ||||||
|  |         state = state.copyWith(iCloudDownloadProgress: progress); | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|       await _backupService.backupAsset( |       await _backupService.backupAsset( | ||||||
|         assetsWillBeBackup, |         assetsWillBeBackup, | ||||||
|         state.cancelToken, |         state.cancelToken, | ||||||
|  |         pmProgressHandler, | ||||||
|         _onAssetUploaded, |         _onAssetUploaded, | ||||||
|         _onUploadProgress, |         _onUploadProgress, | ||||||
|         _onSetCurrentBackupAsset, |         _onSetCurrentBackupAsset, | ||||||
|  | |||||||
| @ -208,10 +208,12 @@ class ManualUploadNotifier extends StateNotifier<ManualUploadState> { | |||||||
|                 state.totalAssetsToUpload == 1; |                 state.totalAssetsToUpload == 1; | ||||||
|         state = |         state = | ||||||
|             state.copyWith(showDetailedNotification: showDetailedNotification); |             state.copyWith(showDetailedNotification: showDetailedNotification); | ||||||
|  |         final pmProgressHandler = PMProgressHandler(); | ||||||
| 
 | 
 | ||||||
|         final bool ok = await ref.read(backupServiceProvider).backupAsset( |         final bool ok = await ref.read(backupServiceProvider).backupAsset( | ||||||
|               allUploadAssets, |               allUploadAssets, | ||||||
|               state.cancelToken, |               state.cancelToken, | ||||||
|  |               pmProgressHandler, | ||||||
|               _onAssetUploaded, |               _onAssetUploaded, | ||||||
|               _onProgress, |               _onProgress, | ||||||
|               _onSetCurrentBackupAsset, |               _onSetCurrentBackupAsset, | ||||||
|  | |||||||
| @ -10,6 +10,8 @@ import 'package:immich_mobile/modules/backup/models/backup_album.model.dart'; | |||||||
| import 'package:immich_mobile/modules/backup/models/current_upload_asset.model.dart'; | import 'package:immich_mobile/modules/backup/models/current_upload_asset.model.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/models/duplicated_asset.model.dart'; | import 'package:immich_mobile/modules/backup/models/duplicated_asset.model.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/models/error_upload_asset.model.dart'; | import 'package:immich_mobile/modules/backup/models/error_upload_asset.model.dart'; | ||||||
|  | import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart'; | ||||||
|  | import 'package:immich_mobile/modules/settings/services/app_settings.service.dart'; | ||||||
| import 'package:immich_mobile/shared/models/store.dart'; | import 'package:immich_mobile/shared/models/store.dart'; | ||||||
| import 'package:immich_mobile/shared/providers/api.provider.dart'; | import 'package:immich_mobile/shared/providers/api.provider.dart'; | ||||||
| import 'package:immich_mobile/shared/providers/db.provider.dart'; | import 'package:immich_mobile/shared/providers/db.provider.dart'; | ||||||
| @ -26,6 +28,7 @@ final backupServiceProvider = Provider( | |||||||
|   (ref) => BackupService( |   (ref) => BackupService( | ||||||
|     ref.watch(apiServiceProvider), |     ref.watch(apiServiceProvider), | ||||||
|     ref.watch(dbProvider), |     ref.watch(dbProvider), | ||||||
|  |     ref.watch(appSettingsServiceProvider), | ||||||
|   ), |   ), | ||||||
| ); | ); | ||||||
| 
 | 
 | ||||||
| @ -34,8 +37,9 @@ class BackupService { | |||||||
|   final ApiService _apiService; |   final ApiService _apiService; | ||||||
|   final Isar _db; |   final Isar _db; | ||||||
|   final Logger _log = Logger("BackupService"); |   final Logger _log = Logger("BackupService"); | ||||||
|  |   final AppSettingsService _appSetting; | ||||||
| 
 | 
 | ||||||
|   BackupService(this._apiService, this._db); |   BackupService(this._apiService, this._db, this._appSetting); | ||||||
| 
 | 
 | ||||||
|   Future<List<String>?> getDeviceBackupAsset() async { |   Future<List<String>?> getDeviceBackupAsset() async { | ||||||
|     final String deviceId = Store.get(StoreKey.deviceId); |     final String deviceId = Store.get(StoreKey.deviceId); | ||||||
| @ -202,12 +206,16 @@ class BackupService { | |||||||
|   Future<bool> backupAsset( |   Future<bool> backupAsset( | ||||||
|     Iterable<AssetEntity> assetList, |     Iterable<AssetEntity> assetList, | ||||||
|     http.CancellationToken cancelToken, |     http.CancellationToken cancelToken, | ||||||
|  |     PMProgressHandler pmProgressHandler, | ||||||
|     Function(String, String, bool) uploadSuccessCb, |     Function(String, String, bool) uploadSuccessCb, | ||||||
|     Function(int, int) uploadProgressCb, |     Function(int, int) uploadProgressCb, | ||||||
|     Function(CurrentUploadAsset) setCurrentUploadAssetCb, |     Function(CurrentUploadAsset) setCurrentUploadAssetCb, | ||||||
|     Function(ErrorUploadAsset) errorCb, { |     Function(ErrorUploadAsset) errorCb, { | ||||||
|     bool sortAssets = false, |     bool sortAssets = false, | ||||||
|   }) async { |   }) async { | ||||||
|  |     final bool isIgnoreIcloudAssets = | ||||||
|  |         _appSetting.getSetting(AppSettingsEnum.ignoreIcloudAssets); | ||||||
|  | 
 | ||||||
|     if (Platform.isAndroid && |     if (Platform.isAndroid && | ||||||
|         !(await Permission.accessMediaLocation.status).isGranted) { |         !(await Permission.accessMediaLocation.status).isGranted) { | ||||||
|       // double check that permission is granted here, to guard against |       // double check that permission is granted here, to guard against | ||||||
| @ -241,11 +249,35 @@ class BackupService { | |||||||
| 
 | 
 | ||||||
|     for (var entity in assetsToUpload) { |     for (var entity in assetsToUpload) { | ||||||
|       try { |       try { | ||||||
|  |         final isAvailableLocally = await entity.isLocallyAvailable(); | ||||||
|  | 
 | ||||||
|  |         // Handle getting files from iCloud | ||||||
|  |         if (!isAvailableLocally && Platform.isIOS) { | ||||||
|  |           // Skip iCloud assets if the user has disabled this feature | ||||||
|  |           if (isIgnoreIcloudAssets) { | ||||||
|  |             continue; | ||||||
|  |           } | ||||||
|  | 
 | ||||||
|  |           setCurrentUploadAssetCb( | ||||||
|  |             CurrentUploadAsset( | ||||||
|  |               id: entity.id, | ||||||
|  |               fileCreatedAt: entity.createDateTime.year == 1970 | ||||||
|  |                   ? entity.modifiedDateTime | ||||||
|  |                   : entity.createDateTime, | ||||||
|  |               fileName: await entity.titleAsync, | ||||||
|  |               fileType: _getAssetType(entity.type), | ||||||
|  |               iCloudAsset: true, | ||||||
|  |             ), | ||||||
|  |           ); | ||||||
|  | 
 | ||||||
|  |           file = await entity.loadFile(progressHandler: pmProgressHandler); | ||||||
|  |         } else { | ||||||
|           if (entity.type == AssetType.video) { |           if (entity.type == AssetType.video) { | ||||||
|             file = await entity.originFile; |             file = await entity.originFile; | ||||||
|           } else { |           } else { | ||||||
|             file = await entity.originFile.timeout(const Duration(seconds: 5)); |             file = await entity.originFile.timeout(const Duration(seconds: 5)); | ||||||
|           } |           } | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         if (file != null) { |         if (file != null) { | ||||||
|           String originalFileName = await entity.titleAsync; |           String originalFileName = await entity.titleAsync; | ||||||
| @ -286,6 +318,7 @@ class BackupService { | |||||||
|                   : entity.createDateTime, |                   : entity.createDateTime, | ||||||
|               fileName: originalFileName, |               fileName: originalFileName, | ||||||
|               fileType: _getAssetType(entity.type), |               fileType: _getAssetType(entity.type), | ||||||
|  |               iCloudAsset: false, | ||||||
|             ), |             ), | ||||||
|           ); |           ); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,3 +1,5 @@ | |||||||
|  | import 'dart:io'; | ||||||
|  | 
 | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/foundation.dart'; | import 'package:flutter/foundation.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| @ -23,6 +25,8 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget { | |||||||
|     var uploadProgress = !isManualUpload |     var uploadProgress = !isManualUpload | ||||||
|         ? ref.watch(backupProvider).progressInPercentage |         ? ref.watch(backupProvider).progressInPercentage | ||||||
|         : ref.watch(manualUploadProvider).progressInPercentage; |         : ref.watch(manualUploadProvider).progressInPercentage; | ||||||
|  |     var iCloudDownloadProgress = | ||||||
|  |         ref.watch(backupProvider).iCloudDownloadProgress; | ||||||
|     final isShowThumbnail = useState(false); |     final isShowThumbnail = useState(false); | ||||||
| 
 | 
 | ||||||
|     String getAssetCreationDate() { |     String getAssetCreationDate() { | ||||||
| @ -143,6 +147,69 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     buildiCloudDownloadProgerssBar() { | ||||||
|  |       if (asset.iCloudAsset != null && asset.iCloudAsset!) { | ||||||
|  |         return Padding( | ||||||
|  |           padding: const EdgeInsets.only(top: 8.0), | ||||||
|  |           child: Row( | ||||||
|  |             children: [ | ||||||
|  |               SizedBox( | ||||||
|  |                 width: 110, | ||||||
|  |                 child: Text( | ||||||
|  |                   "iCloud Download", | ||||||
|  |                   style: context.textTheme.labelSmall, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |               Expanded( | ||||||
|  |                 child: LinearProgressIndicator( | ||||||
|  |                   minHeight: 10.0, | ||||||
|  |                   value: uploadProgress / 100.0, | ||||||
|  |                   backgroundColor: Colors.grey, | ||||||
|  |                   color: context.primaryColor, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |               Text( | ||||||
|  |                 " ${iCloudDownloadProgress.toStringAsFixed(0)}%", | ||||||
|  |                 style: const TextStyle(fontSize: 12), | ||||||
|  |               ), | ||||||
|  |             ], | ||||||
|  |           ), | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       return const SizedBox(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     buildUploadProgressBar() { | ||||||
|  |       return Padding( | ||||||
|  |         padding: const EdgeInsets.only(top: 8.0), | ||||||
|  |         child: Row( | ||||||
|  |           children: [ | ||||||
|  |             if (asset.iCloudAsset != null && asset.iCloudAsset!) | ||||||
|  |               SizedBox( | ||||||
|  |                 width: 110, | ||||||
|  |                 child: Text( | ||||||
|  |                   "Immich Upload", | ||||||
|  |                   style: context.textTheme.labelSmall, | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             Expanded( | ||||||
|  |               child: LinearProgressIndicator( | ||||||
|  |                 minHeight: 10.0, | ||||||
|  |                 value: uploadProgress / 100.0, | ||||||
|  |                 backgroundColor: Colors.grey, | ||||||
|  |                 color: context.primaryColor, | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |             Text( | ||||||
|  |               " ${uploadProgress.toStringAsFixed(0)}%", | ||||||
|  |               style: const TextStyle(fontSize: 12), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     return FutureBuilder<Uint8List?>( |     return FutureBuilder<Uint8List?>( | ||||||
|       future: buildAssetThumbnail(), |       future: buildAssetThumbnail(), | ||||||
|       builder: (context, thumbnail) => ListTile( |       builder: (context, thumbnail) => ListTile( | ||||||
| @ -197,25 +264,8 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget { | |||||||
|         ), |         ), | ||||||
|         subtitle: Column( |         subtitle: Column( | ||||||
|           children: [ |           children: [ | ||||||
|             Padding( |             if (Platform.isIOS) buildiCloudDownloadProgerssBar(), | ||||||
|               padding: const EdgeInsets.only(top: 8.0), |             buildUploadProgressBar(), | ||||||
|               child: Row( |  | ||||||
|                 children: [ |  | ||||||
|                   Expanded( |  | ||||||
|                     child: LinearProgressIndicator( |  | ||||||
|                       minHeight: 10.0, |  | ||||||
|                       value: uploadProgress / 100.0, |  | ||||||
|                       backgroundColor: Colors.grey, |  | ||||||
|                       color: context.primaryColor, |  | ||||||
|                     ), |  | ||||||
|                   ), |  | ||||||
|                   Text( |  | ||||||
|                     " ${uploadProgress.toStringAsFixed(0)}%", |  | ||||||
|                     style: const TextStyle(fontSize: 12), |  | ||||||
|                   ), |  | ||||||
|                 ], |  | ||||||
|               ), |  | ||||||
|             ), |  | ||||||
|             Padding( |             Padding( | ||||||
|               padding: const EdgeInsets.only(top: 8.0), |               padding: const EdgeInsets.only(top: 8.0), | ||||||
|               child: buildAssetInfoTable(), |               child: buildAssetInfoTable(), | ||||||
|  | |||||||
| @ -1,33 +1,21 @@ | |||||||
| import 'dart:io'; | import 'dart:io'; | ||||||
|  | import 'dart:math'; | ||||||
| 
 | 
 | ||||||
| import 'package:connectivity_plus/connectivity_plus.dart'; |  | ||||||
| import 'package:easy_localization/easy_localization.dart'; | import 'package:easy_localization/easy_localization.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter_hooks/flutter_hooks.dart'; | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
| import 'package:immich_mobile/extensions/build_context_extensions.dart'; | import 'package:immich_mobile/extensions/build_context_extensions.dart'; | ||||||
| import 'package:immich_mobile/modules/album/providers/album.provider.dart'; | import 'package:immich_mobile/modules/album/providers/album.provider.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/background_service/background.service.dart'; |  | ||||||
| import 'package:immich_mobile/modules/backup/providers/error_backup_list.provider.dart'; | import 'package:immich_mobile/modules/backup/providers/error_backup_list.provider.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/providers/ios_background_settings.provider.dart'; | import 'package:immich_mobile/modules/backup/providers/ios_background_settings.provider.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/providers/manual_upload.provider.dart'; | import 'package:immich_mobile/modules/backup/providers/manual_upload.provider.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/services/backup_verification.service.dart'; |  | ||||||
| import 'package:immich_mobile/modules/backup/ui/current_backup_asset_info_box.dart'; | import 'package:immich_mobile/modules/backup/ui/current_backup_asset_info_box.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/ui/ios_debug_info_tile.dart'; |  | ||||||
| import 'package:immich_mobile/modules/backup/models/backup_state.model.dart'; | import 'package:immich_mobile/modules/backup/models/backup_state.model.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/providers/backup.provider.dart'; | import 'package:immich_mobile/modules/backup/providers/backup.provider.dart'; | ||||||
| import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart'; |  | ||||||
| import 'package:immich_mobile/modules/settings/services/app_settings.service.dart'; |  | ||||||
| import 'package:immich_mobile/routing/router.dart'; | import 'package:immich_mobile/routing/router.dart'; | ||||||
| import 'package:immich_mobile/shared/models/asset.dart'; |  | ||||||
| import 'package:immich_mobile/shared/providers/asset.provider.dart'; |  | ||||||
| import 'package:immich_mobile/shared/providers/websocket.provider.dart'; | import 'package:immich_mobile/shared/providers/websocket.provider.dart'; | ||||||
| import 'package:immich_mobile/modules/backup/ui/backup_info_card.dart'; | import 'package:immich_mobile/modules/backup/ui/backup_info_card.dart'; | ||||||
| import 'package:immich_mobile/shared/ui/confirm_dialog.dart'; |  | ||||||
| import 'package:immich_mobile/shared/ui/immich_toast.dart'; |  | ||||||
| import 'package:permission_handler/permission_handler.dart'; |  | ||||||
| import 'package:url_launcher/url_launcher.dart'; |  | ||||||
| import 'package:wakelock_plus/wakelock_plus.dart'; |  | ||||||
| 
 | 
 | ||||||
| class BackupControllerPage extends HookConsumerWidget { | class BackupControllerPage extends HookConsumerWidget { | ||||||
|   const BackupControllerPage({Key? key}) : super(key: key); |   const BackupControllerPage({Key? key}) : super(key: key); | ||||||
| @ -35,14 +23,8 @@ class BackupControllerPage extends HookConsumerWidget { | |||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context, WidgetRef ref) { |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|     BackUpState backupState = ref.watch(backupProvider); |     BackUpState backupState = ref.watch(backupProvider); | ||||||
|     final settings = ref.watch(iOSBackgroundSettingsProvider.notifier).settings; |  | ||||||
|     final settingsService = ref.watch(appSettingsServiceProvider); |  | ||||||
|     final showBackupFix = Platform.isAndroid && |  | ||||||
|         settingsService.getSetting(AppSettingsEnum.advancedTroubleshooting); |  | ||||||
|     final hasAnyAlbum = backupState.selectedBackupAlbums.isNotEmpty; |     final hasAnyAlbum = backupState.selectedBackupAlbums.isNotEmpty; | ||||||
| 
 | 
 | ||||||
|     final appRefreshDisabled = |  | ||||||
|         Platform.isIOS && settings?.appRefreshEnabled != true; |  | ||||||
|     bool hasExclusiveAccess = |     bool hasExclusiveAccess = | ||||||
|         backupState.backupProgress != BackUpProgressEnum.inBackground; |         backupState.backupProgress != BackUpProgressEnum.inBackground; | ||||||
|     bool shouldBackup = backupState.allUniqueAssets.length - |     bool shouldBackup = backupState.allUniqueAssets.length - | ||||||
| @ -51,7 +33,6 @@ class BackupControllerPage extends HookConsumerWidget { | |||||||
|             !hasExclusiveAccess |             !hasExclusiveAccess | ||||||
|         ? false |         ? false | ||||||
|         : true; |         : true; | ||||||
|     final checkInProgress = useState(false); |  | ||||||
| 
 | 
 | ||||||
|     useEffect( |     useEffect( | ||||||
|       () { |       () { | ||||||
| @ -75,426 +56,6 @@ class BackupControllerPage extends HookConsumerWidget { | |||||||
|       [], |       [], | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     Future<void> performDeletion(List<Asset> assets) async { |  | ||||||
|       try { |  | ||||||
|         checkInProgress.value = true; |  | ||||||
|         ImmichToast.show( |  | ||||||
|           context: context, |  | ||||||
|           msg: "Deleting ${assets.length} assets on the server...", |  | ||||||
|         ); |  | ||||||
|         await ref |  | ||||||
|             .read(assetProvider.notifier) |  | ||||||
|             .deleteAssets(assets, force: true); |  | ||||||
|         ImmichToast.show( |  | ||||||
|           context: context, |  | ||||||
|           msg: "Deleted ${assets.length} assets on the server. " |  | ||||||
|               "You can now start a manual backup", |  | ||||||
|           toastType: ToastType.success, |  | ||||||
|         ); |  | ||||||
|       } finally { |  | ||||||
|         checkInProgress.value = false; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void performBackupCheck() async { |  | ||||||
|       try { |  | ||||||
|         checkInProgress.value = true; |  | ||||||
|         if (backupState.allUniqueAssets.length > |  | ||||||
|             backupState.selectedAlbumsBackupAssetsIds.length) { |  | ||||||
|           ImmichToast.show( |  | ||||||
|             context: context, |  | ||||||
|             msg: "Backup all assets before starting this check!", |  | ||||||
|             toastType: ToastType.error, |  | ||||||
|           ); |  | ||||||
|           return; |  | ||||||
|         } |  | ||||||
|         final connection = await Connectivity().checkConnectivity(); |  | ||||||
|         if (connection != ConnectivityResult.wifi) { |  | ||||||
|           ImmichToast.show( |  | ||||||
|             context: context, |  | ||||||
|             msg: "Make sure to be connected to unmetered Wi-Fi", |  | ||||||
|             toastType: ToastType.error, |  | ||||||
|           ); |  | ||||||
|           return; |  | ||||||
|         } |  | ||||||
|         WakelockPlus.enable(); |  | ||||||
|         const limit = 100; |  | ||||||
|         final toDelete = await ref |  | ||||||
|             .read(backupVerificationServiceProvider) |  | ||||||
|             .findWronglyBackedUpAssets(limit: limit); |  | ||||||
|         if (toDelete.isEmpty) { |  | ||||||
|           ImmichToast.show( |  | ||||||
|             context: context, |  | ||||||
|             msg: "Did not find any corrupt asset backups!", |  | ||||||
|             toastType: ToastType.success, |  | ||||||
|           ); |  | ||||||
|         } else { |  | ||||||
|           await showDialog( |  | ||||||
|             context: context, |  | ||||||
|             builder: (context) => ConfirmDialog( |  | ||||||
|               onOk: () => performDeletion(toDelete), |  | ||||||
|               title: "Corrupt backups!", |  | ||||||
|               ok: "Delete", |  | ||||||
|               content: |  | ||||||
|                   "Found ${toDelete.length} (max $limit at once) corrupt asset backups. " |  | ||||||
|                   "Run the check again to find more.\n" |  | ||||||
|                   "Do you want to delete the corrupt asset backups now?", |  | ||||||
|             ), |  | ||||||
|           ); |  | ||||||
|         } |  | ||||||
|       } finally { |  | ||||||
|         WakelockPlus.disable(); |  | ||||||
|         checkInProgress.value = false; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     Widget buildCheckCorruptBackups() { |  | ||||||
|       return ListTile( |  | ||||||
|         leading: Icon( |  | ||||||
|           Icons.warning_rounded, |  | ||||||
|           color: context.primaryColor, |  | ||||||
|         ), |  | ||||||
|         title: const Text( |  | ||||||
|           "Check for corrupt asset backups", |  | ||||||
|           style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14), |  | ||||||
|         ), |  | ||||||
|         isThreeLine: true, |  | ||||||
|         subtitle: Column( |  | ||||||
|           crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|           children: [ |  | ||||||
|             const Text("Run this check only over Wi-Fi and once all assets " |  | ||||||
|                 "have been backed-up. The procedure might take a few minutes."), |  | ||||||
|             ElevatedButton( |  | ||||||
|               onPressed: checkInProgress.value ? null : performBackupCheck, |  | ||||||
|               child: checkInProgress.value |  | ||||||
|                   ? const CircularProgressIndicator() |  | ||||||
|                   : const Text("Perform check"), |  | ||||||
|             ), |  | ||||||
|           ], |  | ||||||
|         ), |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ListTile buildAutoBackupController() { |  | ||||||
|       final isAutoBackup = backupState.autoBackup; |  | ||||||
|       final backUpOption = isAutoBackup |  | ||||||
|           ? "backup_controller_page_status_on".tr() |  | ||||||
|           : "backup_controller_page_status_off".tr(); |  | ||||||
|       final backupBtnText = isAutoBackup |  | ||||||
|           ? "backup_controller_page_turn_off".tr() |  | ||||||
|           : "backup_controller_page_turn_on".tr(); |  | ||||||
|       return ListTile( |  | ||||||
|         isThreeLine: true, |  | ||||||
|         leading: isAutoBackup |  | ||||||
|             ? Icon( |  | ||||||
|                 Icons.cloud_done_rounded, |  | ||||||
|                 color: context.primaryColor, |  | ||||||
|               ) |  | ||||||
|             : const Icon(Icons.cloud_off_rounded), |  | ||||||
|         title: Text( |  | ||||||
|           backUpOption, |  | ||||||
|           style: context.textTheme.titleSmall, |  | ||||||
|         ), |  | ||||||
|         subtitle: Padding( |  | ||||||
|           padding: const EdgeInsets.symmetric(vertical: 8.0), |  | ||||||
|           child: Column( |  | ||||||
|             crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|             children: [ |  | ||||||
|               if (!isAutoBackup) |  | ||||||
|                 const Text( |  | ||||||
|                   "backup_controller_page_desc_backup", |  | ||||||
|                   style: TextStyle(fontSize: 14), |  | ||||||
|                 ).tr(), |  | ||||||
|               Padding( |  | ||||||
|                 padding: const EdgeInsets.only(top: 8.0), |  | ||||||
|                 child: ElevatedButton( |  | ||||||
|                   onPressed: () => ref |  | ||||||
|                       .read(backupProvider.notifier) |  | ||||||
|                       .setAutoBackup(!isAutoBackup), |  | ||||||
|                   child: Text( |  | ||||||
|                     backupBtnText, |  | ||||||
|                     style: context.textTheme.labelLarge?.copyWith( |  | ||||||
|                       color: context.isDarkTheme ? Colors.black : Colors.white, |  | ||||||
|                     ), |  | ||||||
|                   ), |  | ||||||
|                 ), |  | ||||||
|               ), |  | ||||||
|             ], |  | ||||||
|           ), |  | ||||||
|         ), |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void showErrorToUser(String msg) { |  | ||||||
|       final snackBar = SnackBar( |  | ||||||
|         content: Text( |  | ||||||
|           msg.tr(), |  | ||||||
|           style: context.textTheme.bodyLarge?.copyWith( |  | ||||||
|             color: context.primaryColor, |  | ||||||
|           ), |  | ||||||
|         ), |  | ||||||
|         backgroundColor: Colors.red, |  | ||||||
|       ); |  | ||||||
|       ScaffoldMessenger.of(context).showSnackBar(snackBar); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void showBatteryOptimizationInfoToUser() { |  | ||||||
|       showDialog<void>( |  | ||||||
|         context: context, |  | ||||||
|         barrierDismissible: false, |  | ||||||
|         builder: (BuildContext context) { |  | ||||||
|           return AlertDialog( |  | ||||||
|             title: const Text( |  | ||||||
|               'backup_controller_page_background_battery_info_title', |  | ||||||
|             ).tr(), |  | ||||||
|             content: SingleChildScrollView( |  | ||||||
|               child: const Text( |  | ||||||
|                 'backup_controller_page_background_battery_info_message', |  | ||||||
|               ).tr(), |  | ||||||
|             ), |  | ||||||
|             actions: [ |  | ||||||
|               ElevatedButton( |  | ||||||
|                 onPressed: () => launchUrl( |  | ||||||
|                   Uri.parse('https://dontkillmyapp.com'), |  | ||||||
|                   mode: LaunchMode.externalApplication, |  | ||||||
|                 ), |  | ||||||
|                 child: const Text( |  | ||||||
|                   "backup_controller_page_background_battery_info_link", |  | ||||||
|                   style: TextStyle(fontWeight: FontWeight.bold, fontSize: 12), |  | ||||||
|                 ).tr(), |  | ||||||
|               ), |  | ||||||
|               ElevatedButton( |  | ||||||
|                 child: const Text( |  | ||||||
|                   'backup_controller_page_background_battery_info_ok', |  | ||||||
|                   style: TextStyle(fontWeight: FontWeight.bold, fontSize: 12), |  | ||||||
|                 ).tr(), |  | ||||||
|                 onPressed: () { |  | ||||||
|                   context.pop(); |  | ||||||
|                 }, |  | ||||||
|               ), |  | ||||||
|             ], |  | ||||||
|           ); |  | ||||||
|         }, |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     Widget buildBackgroundBackupController() { |  | ||||||
|       final bool isBackgroundEnabled = backupState.backgroundBackup; |  | ||||||
|       final bool isWifiRequired = backupState.backupRequireWifi; |  | ||||||
|       final bool isChargingRequired = backupState.backupRequireCharging; |  | ||||||
|       final Color activeColor = context.primaryColor; |  | ||||||
| 
 |  | ||||||
|       String formatBackupDelaySliderValue(double v) { |  | ||||||
|         if (v == 0.0) { |  | ||||||
|           return 'setting_notifications_notify_seconds'.tr(args: const ['5']); |  | ||||||
|         } else if (v == 1.0) { |  | ||||||
|           return 'setting_notifications_notify_seconds'.tr(args: const ['30']); |  | ||||||
|         } else if (v == 2.0) { |  | ||||||
|           return 'setting_notifications_notify_minutes'.tr(args: const ['2']); |  | ||||||
|         } else { |  | ||||||
|           return 'setting_notifications_notify_minutes'.tr(args: const ['10']); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       int backupDelayToMilliseconds(double v) { |  | ||||||
|         if (v == 0.0) { |  | ||||||
|           return 5000; |  | ||||||
|         } else if (v == 1.0) { |  | ||||||
|           return 30000; |  | ||||||
|         } else if (v == 2.0) { |  | ||||||
|           return 120000; |  | ||||||
|         } else { |  | ||||||
|           return 600000; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       double backupDelayToSliderValue(int ms) { |  | ||||||
|         if (ms == 5000) { |  | ||||||
|           return 0.0; |  | ||||||
|         } else if (ms == 30000) { |  | ||||||
|           return 1.0; |  | ||||||
|         } else if (ms == 120000) { |  | ||||||
|           return 2.0; |  | ||||||
|         } else { |  | ||||||
|           return 3.0; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       final triggerDelay = |  | ||||||
|           useState(backupDelayToSliderValue(backupState.backupTriggerDelay)); |  | ||||||
| 
 |  | ||||||
|       return Column( |  | ||||||
|         children: [ |  | ||||||
|           ListTile( |  | ||||||
|             isThreeLine: true, |  | ||||||
|             leading: isBackgroundEnabled |  | ||||||
|                 ? Icon( |  | ||||||
|                     Icons.cloud_sync_rounded, |  | ||||||
|                     color: activeColor, |  | ||||||
|                   ) |  | ||||||
|                 : const Icon(Icons.cloud_sync_rounded), |  | ||||||
|             title: Text( |  | ||||||
|               isBackgroundEnabled |  | ||||||
|                   ? "backup_controller_page_background_is_on" |  | ||||||
|                   : "backup_controller_page_background_is_off", |  | ||||||
|               style: context.textTheme.titleSmall, |  | ||||||
|             ).tr(), |  | ||||||
|             subtitle: Column( |  | ||||||
|               crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|               children: [ |  | ||||||
|                 if (!isBackgroundEnabled) |  | ||||||
|                   Padding( |  | ||||||
|                     padding: const EdgeInsets.symmetric(vertical: 8.0), |  | ||||||
|                     child: const Text( |  | ||||||
|                       "backup_controller_page_background_description", |  | ||||||
|                     ).tr(), |  | ||||||
|                   ), |  | ||||||
|                 if (isBackgroundEnabled && Platform.isAndroid) |  | ||||||
|                   SwitchListTile.adaptive( |  | ||||||
|                     title: const Text("backup_controller_page_background_wifi") |  | ||||||
|                         .tr(), |  | ||||||
|                     secondary: Icon( |  | ||||||
|                       Icons.wifi, |  | ||||||
|                       color: isWifiRequired ? activeColor : null, |  | ||||||
|                     ), |  | ||||||
|                     dense: true, |  | ||||||
|                     activeColor: activeColor, |  | ||||||
|                     value: isWifiRequired, |  | ||||||
|                     onChanged: (isChecked) => ref |  | ||||||
|                         .read(backupProvider.notifier) |  | ||||||
|                         .configureBackgroundBackup( |  | ||||||
|                           requireWifi: isChecked, |  | ||||||
|                           onError: showErrorToUser, |  | ||||||
|                           onBatteryInfo: showBatteryOptimizationInfoToUser, |  | ||||||
|                         ), |  | ||||||
|                   ), |  | ||||||
|                 if (isBackgroundEnabled) |  | ||||||
|                   SwitchListTile.adaptive( |  | ||||||
|                     title: |  | ||||||
|                         const Text("backup_controller_page_background_charging") |  | ||||||
|                             .tr(), |  | ||||||
|                     secondary: Icon( |  | ||||||
|                       Icons.charging_station, |  | ||||||
|                       color: isChargingRequired ? activeColor : null, |  | ||||||
|                     ), |  | ||||||
|                     dense: true, |  | ||||||
|                     activeColor: activeColor, |  | ||||||
|                     value: isChargingRequired, |  | ||||||
|                     onChanged: (isChecked) => ref |  | ||||||
|                         .read(backupProvider.notifier) |  | ||||||
|                         .configureBackgroundBackup( |  | ||||||
|                           requireCharging: isChecked, |  | ||||||
|                           onError: showErrorToUser, |  | ||||||
|                           onBatteryInfo: showBatteryOptimizationInfoToUser, |  | ||||||
|                         ), |  | ||||||
|                   ), |  | ||||||
|                 if (isBackgroundEnabled && Platform.isAndroid) |  | ||||||
|                   ListTile( |  | ||||||
|                     isThreeLine: false, |  | ||||||
|                     dense: true, |  | ||||||
|                     title: const Text( |  | ||||||
|                       'backup_controller_page_background_delay', |  | ||||||
|                       style: TextStyle( |  | ||||||
|                         fontWeight: FontWeight.bold, |  | ||||||
|                       ), |  | ||||||
|                     ).tr( |  | ||||||
|                       args: [formatBackupDelaySliderValue(triggerDelay.value)], |  | ||||||
|                     ), |  | ||||||
|                     subtitle: Slider( |  | ||||||
|                       value: triggerDelay.value, |  | ||||||
|                       onChanged: (double v) => triggerDelay.value = v, |  | ||||||
|                       onChangeEnd: (double v) => ref |  | ||||||
|                           .read(backupProvider.notifier) |  | ||||||
|                           .configureBackgroundBackup( |  | ||||||
|                             triggerDelay: backupDelayToMilliseconds(v), |  | ||||||
|                             onError: showErrorToUser, |  | ||||||
|                             onBatteryInfo: showBatteryOptimizationInfoToUser, |  | ||||||
|                           ), |  | ||||||
|                       max: 3.0, |  | ||||||
|                       divisions: 3, |  | ||||||
|                       label: formatBackupDelaySliderValue(triggerDelay.value), |  | ||||||
|                       activeColor: context.primaryColor, |  | ||||||
|                     ), |  | ||||||
|                   ), |  | ||||||
|                 ElevatedButton( |  | ||||||
|                   onPressed: () => ref |  | ||||||
|                       .read(backupProvider.notifier) |  | ||||||
|                       .configureBackgroundBackup( |  | ||||||
|                         enabled: !isBackgroundEnabled, |  | ||||||
|                         onError: showErrorToUser, |  | ||||||
|                         onBatteryInfo: showBatteryOptimizationInfoToUser, |  | ||||||
|                       ), |  | ||||||
|                   child: Text( |  | ||||||
|                     isBackgroundEnabled |  | ||||||
|                         ? "backup_controller_page_background_turn_off" |  | ||||||
|                         : "backup_controller_page_background_turn_on", |  | ||||||
|                     style: context.textTheme.labelLarge?.copyWith( |  | ||||||
|                       color: context.isDarkTheme ? Colors.black : Colors.white, |  | ||||||
|                     ), |  | ||||||
|                   ).tr(), |  | ||||||
|                 ), |  | ||||||
|               ], |  | ||||||
|             ), |  | ||||||
|           ), |  | ||||||
|           if (isBackgroundEnabled && Platform.isIOS) |  | ||||||
|             FutureBuilder( |  | ||||||
|               future: ref |  | ||||||
|                   .read(backgroundServiceProvider) |  | ||||||
|                   .getIOSBackgroundAppRefreshEnabled(), |  | ||||||
|               builder: (context, snapshot) { |  | ||||||
|                 final enabled = snapshot.data; |  | ||||||
|                 // If it's not enabled, show them some kind of alert that says |  | ||||||
|                 // background refresh is not enabled |  | ||||||
|                 if (enabled != null && !enabled) {} |  | ||||||
|                 // If it's enabled, no need to bother them |  | ||||||
|                 return Container(); |  | ||||||
|               }, |  | ||||||
|             ), |  | ||||||
|           if (Platform.isIOS && isBackgroundEnabled && settings != null) |  | ||||||
|             IosDebugInfoTile( |  | ||||||
|               settings: settings, |  | ||||||
|             ), |  | ||||||
|         ], |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     Widget buildBackgroundAppRefreshWarning() { |  | ||||||
|       return ListTile( |  | ||||||
|         isThreeLine: true, |  | ||||||
|         leading: const Icon( |  | ||||||
|           Icons.task_outlined, |  | ||||||
|         ), |  | ||||||
|         title: const Text( |  | ||||||
|           'backup_controller_page_background_app_refresh_disabled_title', |  | ||||||
|           style: TextStyle( |  | ||||||
|             fontWeight: FontWeight.bold, |  | ||||||
|             fontSize: 14, |  | ||||||
|           ), |  | ||||||
|         ).tr(), |  | ||||||
|         subtitle: Column( |  | ||||||
|           crossAxisAlignment: CrossAxisAlignment.start, |  | ||||||
|           children: [ |  | ||||||
|             Padding( |  | ||||||
|               padding: const EdgeInsets.symmetric(vertical: 8.0), |  | ||||||
|               child: const Text( |  | ||||||
|                 'backup_controller_page_background_app_refresh_disabled_content', |  | ||||||
|               ).tr(), |  | ||||||
|             ), |  | ||||||
|             ElevatedButton( |  | ||||||
|               onPressed: () => openAppSettings(), |  | ||||||
|               child: const Text( |  | ||||||
|                 'backup_controller_page_background_app_refresh_enable_button_text', |  | ||||||
|                 style: TextStyle( |  | ||||||
|                   fontWeight: FontWeight.bold, |  | ||||||
|                   fontSize: 12, |  | ||||||
|                 ), |  | ||||||
|               ).tr(), |  | ||||||
|             ), |  | ||||||
|           ], |  | ||||||
|         ), |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     Widget buildSelectedAlbumName() { |     Widget buildSelectedAlbumName() { | ||||||
|       var text = "backup_controller_page_backup_selected".tr(); |       var text = "backup_controller_page_backup_selected".tr(); | ||||||
|       var albums = ref.watch(backupProvider).selectedBackupAlbums; |       var albums = ref.watch(backupProvider).selectedBackupAlbums; | ||||||
| @ -688,6 +249,18 @@ class BackupControllerPage extends HookConsumerWidget { | |||||||
|             Icons.arrow_back_ios_rounded, |             Icons.arrow_back_ios_rounded, | ||||||
|           ), |           ), | ||||||
|         ), |         ), | ||||||
|  |         actions: [ | ||||||
|  |           Padding( | ||||||
|  |             padding: const EdgeInsets.only(right: 8.0), | ||||||
|  |             child: IconButton( | ||||||
|  |               onPressed: () => context.autoPush(const BackupOptionsRoute()), | ||||||
|  |               splashRadius: 24, | ||||||
|  |               icon: const Icon( | ||||||
|  |                 Icons.settings_outlined, | ||||||
|  |               ), | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |         ], | ||||||
|       ), |       ), | ||||||
|       body: Padding( |       body: Padding( | ||||||
|         padding: const EdgeInsets.only(left: 16.0, right: 16, bottom: 32), |         padding: const EdgeInsets.only(left: 16.0, right: 16, bottom: 32), | ||||||
| @ -715,22 +288,9 @@ class BackupControllerPage extends HookConsumerWidget { | |||||||
|                     subtitle: "backup_controller_page_remainder_sub".tr(), |                     subtitle: "backup_controller_page_remainder_sub".tr(), | ||||||
|                     info: ref.watch(backupProvider).availableAlbums.isEmpty |                     info: ref.watch(backupProvider).availableAlbums.isEmpty | ||||||
|                         ? "..." |                         ? "..." | ||||||
|                         : "${backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length}", |                         : "${max(0, backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length)}", | ||||||
|                   ), |                   ), | ||||||
|                   const Divider(), |                   const Divider(), | ||||||
|                   buildAutoBackupController(), |  | ||||||
|                   const Divider(), |  | ||||||
|                   AnimatedSwitcher( |  | ||||||
|                     duration: const Duration(milliseconds: 500), |  | ||||||
|                     child: Platform.isIOS |  | ||||||
|                         ? (appRefreshDisabled |  | ||||||
|                             ? buildBackgroundAppRefreshWarning() |  | ||||||
|                             : buildBackgroundBackupController()) |  | ||||||
|                         : buildBackgroundBackupController(), |  | ||||||
|                   ), |  | ||||||
|                   if (showBackupFix) const Divider(), |  | ||||||
|                   if (showBackupFix) buildCheckCorruptBackups(), |  | ||||||
|                   const Divider(), |  | ||||||
|                   const CurrentUploadingAssetInfoBox(), |                   const CurrentUploadingAssetInfoBox(), | ||||||
|                   if (!hasExclusiveAccess) buildBackgroundBackupInfo(), |                   if (!hasExclusiveAccess) buildBackgroundBackupInfo(), | ||||||
|                   buildBackupButton(), |                   buildBackupButton(), | ||||||
|  | |||||||
							
								
								
									
										521
									
								
								mobile/lib/modules/backup/views/backup_options_page.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										521
									
								
								mobile/lib/modules/backup/views/backup_options_page.dart
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,521 @@ | |||||||
|  | import 'dart:io'; | ||||||
|  | 
 | ||||||
|  | import 'package:connectivity_plus/connectivity_plus.dart'; | ||||||
|  | import 'package:easy_localization/easy_localization.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:flutter_hooks/flutter_hooks.dart'; | ||||||
|  | import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||||
|  | import 'package:immich_mobile/extensions/build_context_extensions.dart'; | ||||||
|  | import 'package:immich_mobile/modules/backup/background_service/background.service.dart'; | ||||||
|  | import 'package:immich_mobile/modules/backup/models/backup_state.model.dart'; | ||||||
|  | import 'package:immich_mobile/modules/backup/providers/backup.provider.dart'; | ||||||
|  | import 'package:immich_mobile/modules/backup/providers/ios_background_settings.provider.dart'; | ||||||
|  | import 'package:immich_mobile/modules/backup/services/backup_verification.service.dart'; | ||||||
|  | import 'package:immich_mobile/modules/backup/ui/ios_debug_info_tile.dart'; | ||||||
|  | import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart'; | ||||||
|  | import 'package:immich_mobile/modules/settings/services/app_settings.service.dart'; | ||||||
|  | import 'package:immich_mobile/shared/models/asset.dart'; | ||||||
|  | import 'package:immich_mobile/shared/providers/asset.provider.dart'; | ||||||
|  | import 'package:immich_mobile/shared/ui/confirm_dialog.dart'; | ||||||
|  | import 'package:immich_mobile/shared/ui/immich_toast.dart'; | ||||||
|  | import 'package:permission_handler/permission_handler.dart'; | ||||||
|  | import 'package:url_launcher/url_launcher.dart'; | ||||||
|  | import 'package:wakelock_plus/wakelock_plus.dart'; | ||||||
|  | 
 | ||||||
|  | class BackupOptionsPage extends HookConsumerWidget { | ||||||
|  |   const BackupOptionsPage({Key? key}) : super(key: key); | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context, WidgetRef ref) { | ||||||
|  |     BackUpState backupState = ref.watch(backupProvider); | ||||||
|  |     final settings = ref.watch(iOSBackgroundSettingsProvider.notifier).settings; | ||||||
|  |     final settingsService = ref.watch(appSettingsServiceProvider); | ||||||
|  |     final showBackupFix = Platform.isAndroid && | ||||||
|  |         settingsService.getSetting(AppSettingsEnum.advancedTroubleshooting); | ||||||
|  |     final ignoreIcloudAssets = useState( | ||||||
|  |       settingsService.getSetting(AppSettingsEnum.ignoreIcloudAssets), | ||||||
|  |     ); | ||||||
|  |     final appRefreshDisabled = | ||||||
|  |         Platform.isIOS && settings?.appRefreshEnabled != true; | ||||||
|  |     final checkInProgress = useState(false); | ||||||
|  | 
 | ||||||
|  |     Future<void> performDeletion(List<Asset> assets) async { | ||||||
|  |       try { | ||||||
|  |         checkInProgress.value = true; | ||||||
|  |         ImmichToast.show( | ||||||
|  |           context: context, | ||||||
|  |           msg: "Deleting ${assets.length} assets on the server...", | ||||||
|  |         ); | ||||||
|  |         await ref | ||||||
|  |             .read(assetProvider.notifier) | ||||||
|  |             .deleteAssets(assets, force: true); | ||||||
|  |         ImmichToast.show( | ||||||
|  |           context: context, | ||||||
|  |           msg: "Deleted ${assets.length} assets on the server. " | ||||||
|  |               "You can now start a manual backup", | ||||||
|  |           toastType: ToastType.success, | ||||||
|  |         ); | ||||||
|  |       } finally { | ||||||
|  |         checkInProgress.value = false; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void performBackupCheck() async { | ||||||
|  |       try { | ||||||
|  |         checkInProgress.value = true; | ||||||
|  |         if (backupState.allUniqueAssets.length > | ||||||
|  |             backupState.selectedAlbumsBackupAssetsIds.length) { | ||||||
|  |           ImmichToast.show( | ||||||
|  |             context: context, | ||||||
|  |             msg: "Backup all assets before starting this check!", | ||||||
|  |             toastType: ToastType.error, | ||||||
|  |           ); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |         final connection = await Connectivity().checkConnectivity(); | ||||||
|  |         if (connection != ConnectivityResult.wifi) { | ||||||
|  |           ImmichToast.show( | ||||||
|  |             context: context, | ||||||
|  |             msg: "Make sure to be connected to unmetered Wi-Fi", | ||||||
|  |             toastType: ToastType.error, | ||||||
|  |           ); | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |         WakelockPlus.enable(); | ||||||
|  |         const limit = 100; | ||||||
|  |         final toDelete = await ref | ||||||
|  |             .read(backupVerificationServiceProvider) | ||||||
|  |             .findWronglyBackedUpAssets(limit: limit); | ||||||
|  |         if (toDelete.isEmpty) { | ||||||
|  |           ImmichToast.show( | ||||||
|  |             context: context, | ||||||
|  |             msg: "Did not find any corrupt asset backups!", | ||||||
|  |             toastType: ToastType.success, | ||||||
|  |           ); | ||||||
|  |         } else { | ||||||
|  |           await showDialog( | ||||||
|  |             context: context, | ||||||
|  |             builder: (context) => ConfirmDialog( | ||||||
|  |               onOk: () => performDeletion(toDelete), | ||||||
|  |               title: "Corrupt backups!", | ||||||
|  |               ok: "Delete", | ||||||
|  |               content: | ||||||
|  |                   "Found ${toDelete.length} (max $limit at once) corrupt asset backups. " | ||||||
|  |                   "Run the check again to find more.\n" | ||||||
|  |                   "Do you want to delete the corrupt asset backups now?", | ||||||
|  |             ), | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |       } finally { | ||||||
|  |         WakelockPlus.disable(); | ||||||
|  |         checkInProgress.value = false; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Widget buildCheckCorruptBackups() { | ||||||
|  |       return ListTile( | ||||||
|  |         leading: Icon( | ||||||
|  |           Icons.warning_rounded, | ||||||
|  |           color: context.primaryColor, | ||||||
|  |         ), | ||||||
|  |         title: const Text( | ||||||
|  |           "Check for corrupt asset backups", | ||||||
|  |           style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14), | ||||||
|  |         ), | ||||||
|  |         isThreeLine: true, | ||||||
|  |         subtitle: Column( | ||||||
|  |           crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |           children: [ | ||||||
|  |             const Text("Run this check only over Wi-Fi and once all assets " | ||||||
|  |                 "have been backed-up. The procedure might take a few minutes."), | ||||||
|  |             ElevatedButton( | ||||||
|  |               onPressed: checkInProgress.value ? null : performBackupCheck, | ||||||
|  |               child: checkInProgress.value | ||||||
|  |                   ? const CircularProgressIndicator() | ||||||
|  |                   : const Text("Perform check"), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void showErrorToUser(String msg) { | ||||||
|  |       final snackBar = SnackBar( | ||||||
|  |         content: Text( | ||||||
|  |           msg.tr(), | ||||||
|  |           style: context.textTheme.bodyLarge?.copyWith( | ||||||
|  |             color: context.primaryColor, | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |         backgroundColor: Colors.red, | ||||||
|  |       ); | ||||||
|  |       ScaffoldMessenger.of(context).showSnackBar(snackBar); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void showBatteryOptimizationInfoToUser() { | ||||||
|  |       showDialog<void>( | ||||||
|  |         context: context, | ||||||
|  |         barrierDismissible: false, | ||||||
|  |         builder: (BuildContext context) { | ||||||
|  |           return AlertDialog( | ||||||
|  |             title: const Text( | ||||||
|  |               'backup_controller_page_background_battery_info_title', | ||||||
|  |             ).tr(), | ||||||
|  |             content: SingleChildScrollView( | ||||||
|  |               child: const Text( | ||||||
|  |                 'backup_controller_page_background_battery_info_message', | ||||||
|  |               ).tr(), | ||||||
|  |             ), | ||||||
|  |             actions: [ | ||||||
|  |               ElevatedButton( | ||||||
|  |                 onPressed: () => launchUrl( | ||||||
|  |                   Uri.parse('https://dontkillmyapp.com'), | ||||||
|  |                   mode: LaunchMode.externalApplication, | ||||||
|  |                 ), | ||||||
|  |                 child: const Text( | ||||||
|  |                   "backup_controller_page_background_battery_info_link", | ||||||
|  |                   style: TextStyle(fontWeight: FontWeight.bold, fontSize: 12), | ||||||
|  |                 ).tr(), | ||||||
|  |               ), | ||||||
|  |               ElevatedButton( | ||||||
|  |                 child: const Text( | ||||||
|  |                   'backup_controller_page_background_battery_info_ok', | ||||||
|  |                   style: TextStyle(fontWeight: FontWeight.bold, fontSize: 12), | ||||||
|  |                 ).tr(), | ||||||
|  |                 onPressed: () { | ||||||
|  |                   context.pop(); | ||||||
|  |                 }, | ||||||
|  |               ), | ||||||
|  |             ], | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Widget buildBackgroundBackupController() { | ||||||
|  |       final bool isBackgroundEnabled = backupState.backgroundBackup; | ||||||
|  |       final bool isWifiRequired = backupState.backupRequireWifi; | ||||||
|  |       final bool isChargingRequired = backupState.backupRequireCharging; | ||||||
|  |       final Color activeColor = context.primaryColor; | ||||||
|  | 
 | ||||||
|  |       String formatBackupDelaySliderValue(double v) { | ||||||
|  |         if (v == 0.0) { | ||||||
|  |           return 'setting_notifications_notify_seconds'.tr(args: const ['5']); | ||||||
|  |         } else if (v == 1.0) { | ||||||
|  |           return 'setting_notifications_notify_seconds'.tr(args: const ['30']); | ||||||
|  |         } else if (v == 2.0) { | ||||||
|  |           return 'setting_notifications_notify_minutes'.tr(args: const ['2']); | ||||||
|  |         } else { | ||||||
|  |           return 'setting_notifications_notify_minutes'.tr(args: const ['10']); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       int backupDelayToMilliseconds(double v) { | ||||||
|  |         if (v == 0.0) { | ||||||
|  |           return 5000; | ||||||
|  |         } else if (v == 1.0) { | ||||||
|  |           return 30000; | ||||||
|  |         } else if (v == 2.0) { | ||||||
|  |           return 120000; | ||||||
|  |         } else { | ||||||
|  |           return 600000; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       double backupDelayToSliderValue(int ms) { | ||||||
|  |         if (ms == 5000) { | ||||||
|  |           return 0.0; | ||||||
|  |         } else if (ms == 30000) { | ||||||
|  |           return 1.0; | ||||||
|  |         } else if (ms == 120000) { | ||||||
|  |           return 2.0; | ||||||
|  |         } else { | ||||||
|  |           return 3.0; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       final triggerDelay = | ||||||
|  |           useState(backupDelayToSliderValue(backupState.backupTriggerDelay)); | ||||||
|  | 
 | ||||||
|  |       return Column( | ||||||
|  |         children: [ | ||||||
|  |           ListTile( | ||||||
|  |             isThreeLine: true, | ||||||
|  |             leading: isBackgroundEnabled | ||||||
|  |                 ? Icon( | ||||||
|  |                     Icons.cloud_sync_rounded, | ||||||
|  |                     color: activeColor, | ||||||
|  |                   ) | ||||||
|  |                 : const Icon(Icons.cloud_sync_rounded), | ||||||
|  |             title: Text( | ||||||
|  |               isBackgroundEnabled | ||||||
|  |                   ? "backup_controller_page_background_is_on" | ||||||
|  |                   : "backup_controller_page_background_is_off", | ||||||
|  |               style: context.textTheme.titleSmall, | ||||||
|  |             ).tr(), | ||||||
|  |             subtitle: Column( | ||||||
|  |               crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |               children: [ | ||||||
|  |                 if (!isBackgroundEnabled) | ||||||
|  |                   Padding( | ||||||
|  |                     padding: const EdgeInsets.symmetric(vertical: 8.0), | ||||||
|  |                     child: const Text( | ||||||
|  |                       "backup_controller_page_background_description", | ||||||
|  |                     ).tr(), | ||||||
|  |                   ), | ||||||
|  |                 if (isBackgroundEnabled && Platform.isAndroid) | ||||||
|  |                   SwitchListTile.adaptive( | ||||||
|  |                     title: const Text("backup_controller_page_background_wifi") | ||||||
|  |                         .tr(), | ||||||
|  |                     secondary: Icon( | ||||||
|  |                       Icons.wifi, | ||||||
|  |                       color: isWifiRequired ? activeColor : null, | ||||||
|  |                     ), | ||||||
|  |                     dense: true, | ||||||
|  |                     activeColor: activeColor, | ||||||
|  |                     value: isWifiRequired, | ||||||
|  |                     onChanged: (isChecked) => ref | ||||||
|  |                         .read(backupProvider.notifier) | ||||||
|  |                         .configureBackgroundBackup( | ||||||
|  |                           requireWifi: isChecked, | ||||||
|  |                           onError: showErrorToUser, | ||||||
|  |                           onBatteryInfo: showBatteryOptimizationInfoToUser, | ||||||
|  |                         ), | ||||||
|  |                   ), | ||||||
|  |                 if (isBackgroundEnabled) | ||||||
|  |                   SwitchListTile.adaptive( | ||||||
|  |                     title: | ||||||
|  |                         const Text("backup_controller_page_background_charging") | ||||||
|  |                             .tr(), | ||||||
|  |                     secondary: Icon( | ||||||
|  |                       Icons.charging_station, | ||||||
|  |                       color: isChargingRequired ? activeColor : null, | ||||||
|  |                     ), | ||||||
|  |                     dense: true, | ||||||
|  |                     activeColor: activeColor, | ||||||
|  |                     value: isChargingRequired, | ||||||
|  |                     onChanged: (isChecked) => ref | ||||||
|  |                         .read(backupProvider.notifier) | ||||||
|  |                         .configureBackgroundBackup( | ||||||
|  |                           requireCharging: isChecked, | ||||||
|  |                           onError: showErrorToUser, | ||||||
|  |                           onBatteryInfo: showBatteryOptimizationInfoToUser, | ||||||
|  |                         ), | ||||||
|  |                   ), | ||||||
|  |                 if (isBackgroundEnabled && Platform.isAndroid) | ||||||
|  |                   ListTile( | ||||||
|  |                     isThreeLine: false, | ||||||
|  |                     dense: true, | ||||||
|  |                     title: const Text( | ||||||
|  |                       'backup_controller_page_background_delay', | ||||||
|  |                       style: TextStyle( | ||||||
|  |                         fontWeight: FontWeight.bold, | ||||||
|  |                       ), | ||||||
|  |                     ).tr( | ||||||
|  |                       args: [formatBackupDelaySliderValue(triggerDelay.value)], | ||||||
|  |                     ), | ||||||
|  |                     subtitle: Slider( | ||||||
|  |                       value: triggerDelay.value, | ||||||
|  |                       onChanged: (double v) => triggerDelay.value = v, | ||||||
|  |                       onChangeEnd: (double v) => ref | ||||||
|  |                           .read(backupProvider.notifier) | ||||||
|  |                           .configureBackgroundBackup( | ||||||
|  |                             triggerDelay: backupDelayToMilliseconds(v), | ||||||
|  |                             onError: showErrorToUser, | ||||||
|  |                             onBatteryInfo: showBatteryOptimizationInfoToUser, | ||||||
|  |                           ), | ||||||
|  |                       max: 3.0, | ||||||
|  |                       divisions: 3, | ||||||
|  |                       label: formatBackupDelaySliderValue(triggerDelay.value), | ||||||
|  |                       activeColor: context.primaryColor, | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |                 ElevatedButton( | ||||||
|  |                   onPressed: () => ref | ||||||
|  |                       .read(backupProvider.notifier) | ||||||
|  |                       .configureBackgroundBackup( | ||||||
|  |                         enabled: !isBackgroundEnabled, | ||||||
|  |                         onError: showErrorToUser, | ||||||
|  |                         onBatteryInfo: showBatteryOptimizationInfoToUser, | ||||||
|  |                       ), | ||||||
|  |                   child: Text( | ||||||
|  |                     isBackgroundEnabled | ||||||
|  |                         ? "backup_controller_page_background_turn_off" | ||||||
|  |                         : "backup_controller_page_background_turn_on", | ||||||
|  |                     style: context.textTheme.labelLarge?.copyWith( | ||||||
|  |                       color: context.isDarkTheme ? Colors.black : Colors.white, | ||||||
|  |                     ), | ||||||
|  |                   ).tr(), | ||||||
|  |                 ), | ||||||
|  |               ], | ||||||
|  |             ), | ||||||
|  |           ), | ||||||
|  |           if (isBackgroundEnabled && Platform.isIOS) | ||||||
|  |             FutureBuilder( | ||||||
|  |               future: ref | ||||||
|  |                   .read(backgroundServiceProvider) | ||||||
|  |                   .getIOSBackgroundAppRefreshEnabled(), | ||||||
|  |               builder: (context, snapshot) { | ||||||
|  |                 final enabled = snapshot.data; | ||||||
|  |                 // If it's not enabled, show them some kind of alert that says | ||||||
|  |                 // background refresh is not enabled | ||||||
|  |                 if (enabled != null && !enabled) {} | ||||||
|  |                 // If it's enabled, no need to bother them | ||||||
|  |                 return Container(); | ||||||
|  |               }, | ||||||
|  |             ), | ||||||
|  |           if (Platform.isIOS && isBackgroundEnabled && settings != null) | ||||||
|  |             IosDebugInfoTile( | ||||||
|  |               settings: settings, | ||||||
|  |             ), | ||||||
|  |         ], | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Widget buildBackgroundAppRefreshWarning() { | ||||||
|  |       return ListTile( | ||||||
|  |         isThreeLine: true, | ||||||
|  |         leading: const Icon( | ||||||
|  |           Icons.task_outlined, | ||||||
|  |         ), | ||||||
|  |         title: const Text( | ||||||
|  |           'backup_controller_page_background_app_refresh_disabled_title', | ||||||
|  |           style: TextStyle( | ||||||
|  |             fontWeight: FontWeight.bold, | ||||||
|  |             fontSize: 14, | ||||||
|  |           ), | ||||||
|  |         ).tr(), | ||||||
|  |         subtitle: Column( | ||||||
|  |           crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |           children: [ | ||||||
|  |             Padding( | ||||||
|  |               padding: const EdgeInsets.symmetric(vertical: 8.0), | ||||||
|  |               child: const Text( | ||||||
|  |                 'backup_controller_page_background_app_refresh_disabled_content', | ||||||
|  |               ).tr(), | ||||||
|  |             ), | ||||||
|  |             ElevatedButton( | ||||||
|  |               onPressed: () => openAppSettings(), | ||||||
|  |               child: const Text( | ||||||
|  |                 'backup_controller_page_background_app_refresh_enable_button_text', | ||||||
|  |                 style: TextStyle( | ||||||
|  |                   fontWeight: FontWeight.bold, | ||||||
|  |                   fontSize: 12, | ||||||
|  |                 ), | ||||||
|  |               ).tr(), | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ListTile buildAutoBackupController() { | ||||||
|  |       final isAutoBackup = backupState.autoBackup; | ||||||
|  |       final backUpOption = isAutoBackup | ||||||
|  |           ? "backup_controller_page_status_on".tr() | ||||||
|  |           : "backup_controller_page_status_off".tr(); | ||||||
|  |       final backupBtnText = isAutoBackup | ||||||
|  |           ? "backup_controller_page_turn_off".tr() | ||||||
|  |           : "backup_controller_page_turn_on".tr(); | ||||||
|  |       return ListTile( | ||||||
|  |         isThreeLine: true, | ||||||
|  |         leading: isAutoBackup | ||||||
|  |             ? Icon( | ||||||
|  |                 Icons.cloud_done_rounded, | ||||||
|  |                 color: context.primaryColor, | ||||||
|  |               ) | ||||||
|  |             : const Icon(Icons.cloud_off_rounded), | ||||||
|  |         title: Text( | ||||||
|  |           backUpOption, | ||||||
|  |           style: context.textTheme.titleSmall, | ||||||
|  |         ), | ||||||
|  |         subtitle: Padding( | ||||||
|  |           padding: const EdgeInsets.symmetric(vertical: 8.0), | ||||||
|  |           child: Column( | ||||||
|  |             crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|  |             children: [ | ||||||
|  |               if (!isAutoBackup) | ||||||
|  |                 const Text( | ||||||
|  |                   "backup_controller_page_desc_backup", | ||||||
|  |                 ).tr(), | ||||||
|  |               Padding( | ||||||
|  |                 padding: const EdgeInsets.only(top: 8.0), | ||||||
|  |                 child: ElevatedButton( | ||||||
|  |                   onPressed: () => ref | ||||||
|  |                       .read(backupProvider.notifier) | ||||||
|  |                       .setAutoBackup(!isAutoBackup), | ||||||
|  |                   child: Text( | ||||||
|  |                     backupBtnText, | ||||||
|  |                     style: context.textTheme.labelLarge?.copyWith( | ||||||
|  |                       color: context.isDarkTheme ? Colors.black : Colors.white, | ||||||
|  |                     ), | ||||||
|  |                   ), | ||||||
|  |                 ), | ||||||
|  |               ), | ||||||
|  |             ], | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void switchChanged(bool value) { | ||||||
|  |       settingsService.setSetting(AppSettingsEnum.ignoreIcloudAssets, value); | ||||||
|  |       ignoreIcloudAssets.value = value; | ||||||
|  |       ref.invalidate(appSettingsServiceProvider); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     buildIgnoreIcloudAssetSetting() { | ||||||
|  |       return [ | ||||||
|  |         const Divider(), | ||||||
|  |         SwitchListTile.adaptive( | ||||||
|  |           title: const Text( | ||||||
|  |             "Ignore iCloud photos", | ||||||
|  |             style: TextStyle(fontWeight: FontWeight.bold), | ||||||
|  |           ), | ||||||
|  |           subtitle: const Text( | ||||||
|  |             "Photos that are stored on iCloud will not be uploaded to the Immich server", | ||||||
|  |           ), | ||||||
|  |           value: ignoreIcloudAssets.value, | ||||||
|  |           onChanged: switchChanged, | ||||||
|  |         ), | ||||||
|  |       ]; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return Scaffold( | ||||||
|  |       appBar: AppBar( | ||||||
|  |         elevation: 0, | ||||||
|  |         title: const Text( | ||||||
|  |           "Backup options", | ||||||
|  |         ), | ||||||
|  |         leading: IconButton( | ||||||
|  |           onPressed: () { | ||||||
|  |             context.autoPop(true); | ||||||
|  |           }, | ||||||
|  |           splashRadius: 24, | ||||||
|  |           icon: const Icon( | ||||||
|  |             Icons.arrow_back_ios_rounded, | ||||||
|  |           ), | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|  |       body: Padding( | ||||||
|  |         padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 32.0), | ||||||
|  |         child: ListView( | ||||||
|  |           children: [ | ||||||
|  |             buildAutoBackupController(), | ||||||
|  |             const Divider(), | ||||||
|  |             AnimatedSwitcher( | ||||||
|  |               duration: const Duration(milliseconds: 500), | ||||||
|  |               child: Platform.isIOS | ||||||
|  |                   ? (appRefreshDisabled | ||||||
|  |                       ? buildBackgroundAppRefreshWarning() | ||||||
|  |                       : buildBackgroundBackupController()) | ||||||
|  |                   : buildBackgroundBackupController(), | ||||||
|  |             ), | ||||||
|  |             if (Platform.isIOS) ...buildIgnoreIcloudAssetSetting(), | ||||||
|  |             if (showBackupFix) const Divider(), | ||||||
|  |             if (showBackupFix) buildCheckCorruptBackups(), | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|  |       ), | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @ -51,6 +51,7 @@ enum AppSettingsEnum<T> { | |||||||
|   mapIncludeArchived<bool>(StoreKey.mapIncludeArchived, null, false), |   mapIncludeArchived<bool>(StoreKey.mapIncludeArchived, null, false), | ||||||
|   mapRelativeDate<int>(StoreKey.mapRelativeDate, null, 0), |   mapRelativeDate<int>(StoreKey.mapRelativeDate, null, 0), | ||||||
|   allowSelfSignedSSLCert<bool>(StoreKey.selfSignedCert, null, false), |   allowSelfSignedSSLCert<bool>(StoreKey.selfSignedCert, null, false), | ||||||
|  |   ignoreIcloudAssets<bool>(StoreKey.ignoreIcloudAssets, null, false), | ||||||
|   ; |   ; | ||||||
| 
 | 
 | ||||||
|   const AppSettingsEnum(this.storeKey, this.hiveKey, this.defaultValue); |   const AppSettingsEnum(this.storeKey, this.hiveKey, this.defaultValue); | ||||||
|  | |||||||
| @ -8,6 +8,7 @@ import 'package:immich_mobile/modules/album/views/album_viewer_page.dart'; | |||||||
| import 'package:immich_mobile/modules/album/views/asset_selection_page.dart'; | import 'package:immich_mobile/modules/album/views/asset_selection_page.dart'; | ||||||
| import 'package:immich_mobile/modules/album/views/create_album_page.dart'; | import 'package:immich_mobile/modules/album/views/create_album_page.dart'; | ||||||
| import 'package:immich_mobile/modules/album/views/library_page.dart'; | import 'package:immich_mobile/modules/album/views/library_page.dart'; | ||||||
|  | import 'package:immich_mobile/modules/backup/views/backup_options_page.dart'; | ||||||
| import 'package:immich_mobile/modules/map/ui/map_location_picker.dart'; | import 'package:immich_mobile/modules/map/ui/map_location_picker.dart'; | ||||||
| import 'package:immich_mobile/modules/map/views/map_page.dart'; | import 'package:immich_mobile/modules/map/views/map_page.dart'; | ||||||
| import 'package:immich_mobile/modules/memories/models/memory.dart'; | import 'package:immich_mobile/modules/memories/models/memory.dart'; | ||||||
| @ -178,6 +179,7 @@ part 'router.gr.dart'; | |||||||
|       page: MapLocationPickerPage, |       page: MapLocationPickerPage, | ||||||
|       guards: [AuthGuard, DuplicateGuard], |       guards: [AuthGuard, DuplicateGuard], | ||||||
|     ), |     ), | ||||||
|  |     AutoRoute(page: BackupOptionsPage, guards: [AuthGuard, DuplicateGuard]), | ||||||
|   ], |   ], | ||||||
| ) | ) | ||||||
| class AppRouter extends _$AppRouter { | class AppRouter extends _$AppRouter { | ||||||
|  | |||||||
| @ -371,6 +371,12 @@ class _$AppRouter extends RootStackRouter { | |||||||
|         barrierDismissible: false, |         barrierDismissible: false, | ||||||
|       ); |       ); | ||||||
|     }, |     }, | ||||||
|  |     BackupOptionsRoute.name: (routeData) { | ||||||
|  |       return MaterialPageX<dynamic>( | ||||||
|  |         routeData: routeData, | ||||||
|  |         child: const BackupOptionsPage(), | ||||||
|  |       ); | ||||||
|  |     }, | ||||||
|     HomeRoute.name: (routeData) { |     HomeRoute.name: (routeData) { | ||||||
|       return MaterialPageX<dynamic>( |       return MaterialPageX<dynamic>( | ||||||
|         routeData: routeData, |         routeData: routeData, | ||||||
| @ -723,6 +729,14 @@ class _$AppRouter extends RootStackRouter { | |||||||
|             duplicateGuard, |             duplicateGuard, | ||||||
|           ], |           ], | ||||||
|         ), |         ), | ||||||
|  |         RouteConfig( | ||||||
|  |           BackupOptionsRoute.name, | ||||||
|  |           path: '/backup-options-page', | ||||||
|  |           guards: [ | ||||||
|  |             authGuard, | ||||||
|  |             duplicateGuard, | ||||||
|  |           ], | ||||||
|  |         ), | ||||||
|       ]; |       ]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1664,6 +1678,18 @@ class MapLocationPickerRouteArgs { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// generated route for | ||||||
|  | /// [BackupOptionsPage] | ||||||
|  | class BackupOptionsRoute extends PageRouteInfo<void> { | ||||||
|  |   const BackupOptionsRoute() | ||||||
|  |       : super( | ||||||
|  |           BackupOptionsRoute.name, | ||||||
|  |           path: '/backup-options-page', | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |   static const String name = 'BackupOptionsRoute'; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /// generated route for | /// generated route for | ||||||
| /// [HomePage] | /// [HomePage] | ||||||
| class HomeRoute extends PageRouteInfo<void> { | class HomeRoute extends PageRouteInfo<void> { | ||||||
|  | |||||||
| @ -182,6 +182,7 @@ enum StoreKey<T> { | |||||||
|   mapRelativeDate<int>(119, type: int), |   mapRelativeDate<int>(119, type: int), | ||||||
|   selfSignedCert<bool>(120, type: bool), |   selfSignedCert<bool>(120, type: bool), | ||||||
|   mapIncludeArchived<bool>(121, type: bool), |   mapIncludeArchived<bool>(121, type: bool), | ||||||
|  |   ignoreIcloudAssets<bool>(122, type: bool), | ||||||
|   ; |   ; | ||||||
| 
 | 
 | ||||||
|   const StoreKey( |   const StoreKey( | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user