mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-26 00:02:34 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			348 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:io';
 | |
| import 'dart:math';
 | |
| 
 | |
| import 'package:auto_route/auto_route.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/extensions/theme_extensions.dart';
 | |
| import 'package:immich_mobile/models/backup/backup_state.model.dart';
 | |
| import 'package:immich_mobile/providers/album/album.provider.dart';
 | |
| import 'package:immich_mobile/providers/backup/backup.provider.dart';
 | |
| import 'package:immich_mobile/providers/backup/error_backup_list.provider.dart';
 | |
| import 'package:immich_mobile/providers/backup/ios_background_settings.provider.dart';
 | |
| import 'package:immich_mobile/providers/backup/manual_upload.provider.dart';
 | |
| import 'package:immich_mobile/providers/websocket.provider.dart';
 | |
| import 'package:immich_mobile/routing/router.dart';
 | |
| import 'package:immich_mobile/widgets/backup/backup_info_card.dart';
 | |
| import 'package:immich_mobile/widgets/backup/current_backup_asset_info_box.dart';
 | |
| import 'package:wakelock_plus/wakelock_plus.dart';
 | |
| 
 | |
| @RoutePage()
 | |
| class BackupControllerPage extends HookConsumerWidget {
 | |
|   const BackupControllerPage({super.key});
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context, WidgetRef ref) {
 | |
|     BackUpState backupState = ref.watch(backupProvider);
 | |
|     final hasAnyAlbum = backupState.selectedBackupAlbums.isNotEmpty;
 | |
|     final didGetBackupInfo = useState(false);
 | |
| 
 | |
|     bool hasExclusiveAccess =
 | |
|         backupState.backupProgress != BackUpProgressEnum.inBackground;
 | |
|     bool shouldBackup = backupState.allUniqueAssets.length -
 | |
|                     backupState.selectedAlbumsBackupAssetsIds.length ==
 | |
|                 0 ||
 | |
|             !hasExclusiveAccess
 | |
|         ? false
 | |
|         : true;
 | |
| 
 | |
|     useEffect(
 | |
|       () {
 | |
|         // Update the background settings information just to make sure we
 | |
|         // have the latest, since the platform channel will not update
 | |
|         // automatically
 | |
|         if (Platform.isIOS) {
 | |
|           ref.watch(iOSBackgroundSettingsProvider.notifier).refresh();
 | |
|         }
 | |
| 
 | |
|         ref
 | |
|             .watch(websocketProvider.notifier)
 | |
|             .stopListenToEvent('on_upload_success');
 | |
| 
 | |
|         return () {
 | |
|           WakelockPlus.disable();
 | |
|         };
 | |
|       },
 | |
|       [],
 | |
|     );
 | |
| 
 | |
|     useEffect(
 | |
|       () {
 | |
|         if (backupState.backupProgress == BackUpProgressEnum.idle &&
 | |
|             !didGetBackupInfo.value) {
 | |
|           ref.watch(backupProvider.notifier).getBackupInfo();
 | |
|           didGetBackupInfo.value = true;
 | |
|         }
 | |
|         return null;
 | |
|       },
 | |
|       [backupState.backupProgress],
 | |
|     );
 | |
| 
 | |
|     useEffect(
 | |
|       () {
 | |
|         if (backupState.backupProgress == BackUpProgressEnum.inProgress) {
 | |
|           WakelockPlus.enable();
 | |
|         } else {
 | |
|           WakelockPlus.disable();
 | |
|         }
 | |
| 
 | |
|         return null;
 | |
|       },
 | |
|       [backupState.backupProgress],
 | |
|     );
 | |
| 
 | |
|     Widget buildSelectedAlbumName() {
 | |
|       var text = "backup_controller_page_backup_selected".tr();
 | |
|       var albums = ref.watch(backupProvider).selectedBackupAlbums;
 | |
| 
 | |
|       if (albums.isNotEmpty) {
 | |
|         for (var album in albums) {
 | |
|           if (album.name == "Recent" || album.name == "Recents") {
 | |
|             text += "${album.name} (${'backup_all'.tr()}), ";
 | |
|           } else {
 | |
|             text += "${album.name}, ";
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         return Padding(
 | |
|           padding: const EdgeInsets.only(top: 8.0),
 | |
|           child: Text(
 | |
|             text.trim().substring(0, text.length - 2),
 | |
|             style: context.textTheme.labelLarge?.copyWith(
 | |
|               color: context.primaryColor,
 | |
|             ),
 | |
|           ),
 | |
|         );
 | |
|       } else {
 | |
|         return Padding(
 | |
|           padding: const EdgeInsets.only(top: 8.0),
 | |
|           child: Text(
 | |
|             "backup_controller_page_none_selected".tr(),
 | |
|             style: context.textTheme.labelLarge?.copyWith(
 | |
|               color: context.primaryColor,
 | |
|             ),
 | |
|           ),
 | |
|         );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     Widget buildExcludedAlbumName() {
 | |
|       var text = "backup_controller_page_excluded".tr();
 | |
|       var albums = ref.watch(backupProvider).excludedBackupAlbums;
 | |
| 
 | |
|       if (albums.isNotEmpty) {
 | |
|         for (var album in albums) {
 | |
|           text += "${album.name}, ";
 | |
|         }
 | |
| 
 | |
|         return Padding(
 | |
|           padding: const EdgeInsets.only(top: 8.0),
 | |
|           child: Text(
 | |
|             text.trim().substring(0, text.length - 2),
 | |
|             style: context.textTheme.labelLarge?.copyWith(
 | |
|               color: Colors.red[300],
 | |
|             ),
 | |
|           ),
 | |
|         );
 | |
|       } else {
 | |
|         return const SizedBox();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     buildFolderSelectionTile() {
 | |
|       return Padding(
 | |
|         padding: const EdgeInsets.only(top: 8.0),
 | |
|         child: Card(
 | |
|           shape: RoundedRectangleBorder(
 | |
|             borderRadius: BorderRadius.circular(20),
 | |
|             side: BorderSide(
 | |
|               color: context.colorScheme.outlineVariant,
 | |
|               width: 1,
 | |
|             ),
 | |
|           ),
 | |
|           elevation: 0,
 | |
|           borderOnForeground: false,
 | |
|           child: ListTile(
 | |
|             minVerticalPadding: 18,
 | |
|             title: Text(
 | |
|               "backup_controller_page_albums",
 | |
|               style: context.textTheme.titleMedium,
 | |
|             ).tr(),
 | |
|             subtitle: Padding(
 | |
|               padding: const EdgeInsets.only(top: 8.0),
 | |
|               child: Column(
 | |
|                 crossAxisAlignment: CrossAxisAlignment.start,
 | |
|                 children: [
 | |
|                   Text(
 | |
|                     "backup_controller_page_to_backup",
 | |
|                     style: context.textTheme.bodyMedium?.copyWith(
 | |
|                       color: context.colorScheme.onSurfaceSecondary,
 | |
|                     ),
 | |
|                   ).tr(),
 | |
|                   buildSelectedAlbumName(),
 | |
|                   buildExcludedAlbumName(),
 | |
|                 ],
 | |
|               ),
 | |
|             ),
 | |
|             trailing: ElevatedButton(
 | |
|               onPressed: () async {
 | |
|                 await context.pushRoute(const BackupAlbumSelectionRoute());
 | |
|                 // waited until returning from selection
 | |
|                 await ref
 | |
|                     .read(backupProvider.notifier)
 | |
|                     .backupAlbumSelectionDone();
 | |
|                 // waited until backup albums are stored in DB
 | |
|                 ref.read(albumProvider.notifier).refreshDeviceAlbums();
 | |
|               },
 | |
|               child: const Text(
 | |
|                 "backup_controller_page_select",
 | |
|                 style: TextStyle(
 | |
|                   fontWeight: FontWeight.bold,
 | |
|                 ),
 | |
|               ).tr(),
 | |
|             ),
 | |
|           ),
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     void startBackup() {
 | |
|       ref.watch(errorBackupListProvider.notifier).empty();
 | |
|       if (ref.watch(backupProvider).backupProgress !=
 | |
|           BackUpProgressEnum.inBackground) {
 | |
|         ref.watch(backupProvider.notifier).startBackupProcess();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     Widget buildBackupButton() {
 | |
|       return Padding(
 | |
|         padding: const EdgeInsets.only(
 | |
|           top: 24,
 | |
|         ),
 | |
|         child: Container(
 | |
|           child: backupState.backupProgress == BackUpProgressEnum.inProgress ||
 | |
|                   backupState.backupProgress ==
 | |
|                       BackUpProgressEnum.manualInProgress
 | |
|               ? ElevatedButton(
 | |
|                   style: ElevatedButton.styleFrom(
 | |
|                     foregroundColor: Colors.grey[50],
 | |
|                     backgroundColor: Colors.red[300],
 | |
|                     // padding: const EdgeInsets.all(14),
 | |
|                   ),
 | |
|                   onPressed: () {
 | |
|                     if (backupState.backupProgress ==
 | |
|                         BackUpProgressEnum.manualInProgress) {
 | |
|                       ref.read(manualUploadProvider.notifier).cancelBackup();
 | |
|                     } else {
 | |
|                       ref.read(backupProvider.notifier).cancelBackup();
 | |
|                     }
 | |
|                   },
 | |
|                   child: const Text(
 | |
|                     "backup_controller_page_cancel",
 | |
|                     style: TextStyle(
 | |
|                       fontSize: 14,
 | |
|                       fontWeight: FontWeight.bold,
 | |
|                     ),
 | |
|                   ).tr(),
 | |
|                 )
 | |
|               : ElevatedButton(
 | |
|                   onPressed: shouldBackup ? startBackup : null,
 | |
|                   child: const Text(
 | |
|                     "backup_controller_page_start_backup",
 | |
|                     style: TextStyle(
 | |
|                       fontSize: 16,
 | |
|                       fontWeight: FontWeight.bold,
 | |
|                     ),
 | |
|                   ).tr(),
 | |
|                 ),
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     buildBackgroundBackupInfo() {
 | |
|       return const ListTile(
 | |
|         leading: Icon(Icons.info_outline_rounded),
 | |
|         title: Text(
 | |
|           "Background backup is currently running, cannot start manual backup",
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     buildLoadingIndicator() {
 | |
|       return const Padding(
 | |
|         padding: EdgeInsets.only(top: 42.0),
 | |
|         child: Center(
 | |
|           child: CircularProgressIndicator(),
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     return Scaffold(
 | |
|       appBar: AppBar(
 | |
|         elevation: 0,
 | |
|         title: const Text(
 | |
|           "backup_controller_page_backup",
 | |
|         ).tr(),
 | |
|         leading: IconButton(
 | |
|           onPressed: () {
 | |
|             ref.watch(websocketProvider.notifier).listenUploadEvent();
 | |
|             context.maybePop(true);
 | |
|           },
 | |
|           splashRadius: 24,
 | |
|           icon: const Icon(
 | |
|             Icons.arrow_back_ios_rounded,
 | |
|           ),
 | |
|         ),
 | |
|         actions: [
 | |
|           Padding(
 | |
|             padding: const EdgeInsets.only(right: 8.0),
 | |
|             child: IconButton(
 | |
|               onPressed: () => context.pushRoute(const BackupOptionsRoute()),
 | |
|               splashRadius: 24,
 | |
|               icon: const Icon(
 | |
|                 Icons.settings_outlined,
 | |
|               ),
 | |
|             ),
 | |
|           ),
 | |
|         ],
 | |
|       ),
 | |
|       body: Stack(
 | |
|         children: [
 | |
|           Padding(
 | |
|             padding: const EdgeInsets.only(left: 16.0, right: 16, bottom: 32),
 | |
|             child: ListView(
 | |
|               // crossAxisAlignment: CrossAxisAlignment.start,
 | |
|               children: hasAnyAlbum
 | |
|                   ? [
 | |
|                       buildFolderSelectionTile(),
 | |
|                       BackupInfoCard(
 | |
|                         title: "backup_controller_page_total".tr(),
 | |
|                         subtitle: "backup_controller_page_total_sub".tr(),
 | |
|                         info: ref.watch(backupProvider).availableAlbums.isEmpty
 | |
|                             ? "..."
 | |
|                             : "${backupState.allUniqueAssets.length}",
 | |
|                       ),
 | |
|                       BackupInfoCard(
 | |
|                         title: "backup_controller_page_backup".tr(),
 | |
|                         subtitle: "backup_controller_page_backup_sub".tr(),
 | |
|                         info: ref.watch(backupProvider).availableAlbums.isEmpty
 | |
|                             ? "..."
 | |
|                             : "${backupState.selectedAlbumsBackupAssetsIds.length}",
 | |
|                       ),
 | |
|                       BackupInfoCard(
 | |
|                         title: "backup_controller_page_remainder".tr(),
 | |
|                         subtitle: "backup_controller_page_remainder_sub".tr(),
 | |
|                         info: ref.watch(backupProvider).availableAlbums.isEmpty
 | |
|                             ? "..."
 | |
|                             : "${max(0, backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length)}",
 | |
|                       ),
 | |
|                       const Divider(),
 | |
|                       const CurrentUploadingAssetInfoBox(),
 | |
|                       if (!hasExclusiveAccess) buildBackgroundBackupInfo(),
 | |
|                       buildBackupButton(),
 | |
|                     ]
 | |
|                   : [
 | |
|                       buildFolderSelectionTile(),
 | |
|                       if (!didGetBackupInfo.value) buildLoadingIndicator(),
 | |
|                     ],
 | |
|             ),
 | |
|           ),
 | |
|         ],
 | |
|       ),
 | |
|     );
 | |
|   }
 | |
| }
 |