import 'package:auto_route/auto_route.dart'; import 'package:flutter/material.dart'; import 'package:flutter_hooks/flutter_hooks.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:immich_mobile/modules/backup/providers/error_backup_list.provider.dart'; import 'package:immich_mobile/modules/login/models/authentication_state.model.dart'; import 'package:immich_mobile/modules/backup/models/backup_state.model.dart'; import 'package:immich_mobile/modules/login/providers/authentication.provider.dart'; import 'package:immich_mobile/modules/backup/providers/backup.provider.dart'; import 'package:immich_mobile/routing/router.dart'; import 'package:immich_mobile/shared/providers/websocket.provider.dart'; import 'package:immich_mobile/modules/backup/ui/backup_info_card.dart'; import 'package:intl/intl.dart'; import 'package:percent_indicator/linear_percent_indicator.dart'; class BackupControllerPage extends HookConsumerWidget { const BackupControllerPage({Key? key}) : super(key: key); @override Widget build(BuildContext context, WidgetRef ref) { BackUpState backupState = ref.watch(backupProvider); AuthenticationState authenticationState = ref.watch(authenticationProvider); bool shouldBackup = backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length == 0 ? false : true; useEffect(() { if (backupState.backupProgress != BackUpProgressEnum.inProgress) { ref.watch(backupProvider.notifier).getBackupInfo(); } ref .watch(websocketProvider.notifier) .stopListenToEvent('on_upload_success'); return null; }, []); Widget _buildStorageInformation() { return ListTile( leading: Icon( Icons.storage_rounded, color: Theme.of(context).primaryColor, ), title: const Text( "Server storage", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14), ), subtitle: Padding( padding: const EdgeInsets.only(top: 8.0), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Padding( padding: const EdgeInsets.only(top: 8.0), child: LinearPercentIndicator( padding: const EdgeInsets.symmetric(horizontal: 0, vertical: 0), barRadius: const Radius.circular(2), lineHeight: 10.0, percent: backupState.serverInfo.diskUsagePercentage / 100.0, backgroundColor: Colors.grey, progressColor: Theme.of(context).primaryColor, ), ), Padding( padding: const EdgeInsets.only(top: 12.0), child: Text( '${backupState.serverInfo.diskUse} of ${backupState.serverInfo.diskSize} used'), ), ], ), ), ); } ListTile _buildBackupController() { var backUpOption = authenticationState.deviceInfo.isAutoBackup ? "on" : "off"; var isAutoBackup = authenticationState.deviceInfo.isAutoBackup; var backupBtnText = authenticationState.deviceInfo.isAutoBackup ? "off" : "on"; return ListTile( isThreeLine: true, leading: isAutoBackup ? Icon( Icons.cloud_done_rounded, color: Theme.of(context).primaryColor, ) : const Icon(Icons.cloud_off_rounded), title: Text( "Back up is $backUpOption", style: const TextStyle(fontWeight: FontWeight.bold, fontSize: 14), ), subtitle: Padding( padding: const EdgeInsets.symmetric(vertical: 8.0), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ if (!isAutoBackup) const Text( "Turn on backup to automatically upload new assets to the server.", style: TextStyle(fontSize: 14), ), Padding( padding: const EdgeInsets.only(top: 8.0), child: OutlinedButton( style: OutlinedButton.styleFrom( side: const BorderSide( width: 1, color: Color.fromARGB(255, 220, 220, 220), ), ), onPressed: () { if (isAutoBackup) { ref .read(authenticationProvider.notifier) .setAutoBackup(false); } else { ref .read(authenticationProvider.notifier) .setAutoBackup(true); } }, child: Text("Turn $backupBtnText Backup", style: const TextStyle(fontWeight: FontWeight.bold)), ), ) ], ), ), ); } Widget _buildSelectedAlbumName() { var text = "Selected: "; var albums = ref.watch(backupProvider).selectedBackupAlbums; if (albums.isNotEmpty) { for (var album in albums) { if (album.name == "Recent" || album.name == "Recents") { text += "${album.name} (All), "; } else { text += "${album.name}, "; } } return Padding( padding: const EdgeInsets.only(top: 8.0), child: Text( text.trim().substring(0, text.length - 2), style: TextStyle( color: Theme.of(context).primaryColor, fontSize: 12, fontWeight: FontWeight.bold), ), ); } else { return Padding( padding: const EdgeInsets.only(top: 8.0), child: Text( "None selected", style: TextStyle( color: Theme.of(context).primaryColor, fontSize: 12, fontWeight: FontWeight.bold), ), ); } } Widget _buildExcludedAlbumName() { var text = "Excluded: "; 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: TextStyle( color: Colors.red[300], fontSize: 12, fontWeight: FontWeight.bold), ), ); } else { return const SizedBox(); } } _buildFolderSelectionTile() { return Card( shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(5), // if you need this side: const BorderSide( color: Colors.black12, width: 1, ), ), elevation: 0, borderOnForeground: false, child: ListTile( minVerticalPadding: 15, title: const Text("Backup Albums", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 20)), subtitle: Padding( padding: const EdgeInsets.only(top: 8.0), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ const Text( "Albums to be backed up", style: TextStyle(color: Color(0xFF808080), fontSize: 12), ), _buildSelectedAlbumName(), _buildExcludedAlbumName() ], ), ), trailing: OutlinedButton( style: OutlinedButton.styleFrom( enableFeedback: true, side: const BorderSide( width: 1, color: Color.fromARGB(255, 220, 220, 220), ), ), onPressed: () { AutoRouter.of(context).push(const BackupAlbumSelectionRoute()); }, child: const Padding( padding: EdgeInsets.symmetric( vertical: 16.0, ), child: Text( "Select", style: TextStyle(fontWeight: FontWeight.bold), ), ), ), ), ); } _buildCurrentBackupAssetInfoCard() { return ListTile( leading: Icon( Icons.info_outline_rounded, color: Theme.of(context).primaryColor, ), title: Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ const Text( "Uploading file info", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14), ), if (ref.watch(errorBackupListProvider).isNotEmpty) ActionChip( avatar: Icon( Icons.info, size: 24, color: Colors.red[400], ), elevation: 1, visualDensity: VisualDensity.compact, label: Text( "Failed (${ref.watch(errorBackupListProvider).length})", style: TextStyle( color: Colors.red[400], fontWeight: FontWeight.bold, fontSize: 11, ), ), backgroundColor: Colors.white, onPressed: () { AutoRouter.of(context).push(const FailedBackupStatusRoute()); }, ), ], ), subtitle: Column( children: [ Padding( padding: const EdgeInsets.only(top: 8.0), child: LinearPercentIndicator( padding: const EdgeInsets.symmetric(horizontal: 0, vertical: 0), barRadius: const Radius.circular(2), lineHeight: 10.0, trailing: Text( " ${backupState.progressInPercentage.toStringAsFixed(0)}%", style: const TextStyle(fontSize: 12), ), percent: backupState.progressInPercentage / 100.0, backgroundColor: Colors.grey, progressColor: Theme.of(context).primaryColor, ), ), Padding( padding: const EdgeInsets.only(top: 8.0), child: Table( border: TableBorder.all( color: Colors.black12, width: 1, ), children: [ TableRow( decoration: BoxDecoration( color: Colors.grey[100], ), children: [ TableCell( verticalAlignment: TableCellVerticalAlignment.middle, child: Padding( padding: const EdgeInsets.all(6.0), child: Text( 'File name: ${backupState.currentUploadAsset.fileName} [${backupState.currentUploadAsset.fileType.toLowerCase()}]', style: const TextStyle( fontWeight: FontWeight.bold, fontSize: 10.0), ), ), ), ], ), TableRow( decoration: BoxDecoration( color: Colors.grey[200], ), children: [ TableCell( verticalAlignment: TableCellVerticalAlignment.middle, child: Padding( padding: const EdgeInsets.all(6.0), child: Text( "Created on: ${DateFormat.yMMMMd('en_US').format( DateTime.parse( backupState.currentUploadAsset.createdAt .toString(), ), )}", style: const TextStyle( fontWeight: FontWeight.bold, fontSize: 10.0), ), ), ), ], ), TableRow( decoration: BoxDecoration( color: Colors.grey[100], ), children: [ TableCell( child: Padding( padding: const EdgeInsets.all(6.0), child: Text( "ID: ${backupState.currentUploadAsset.id}", style: const TextStyle( fontWeight: FontWeight.bold, fontSize: 10.0, ), ), ), ), ], ), ], ), ), ], ), ); } void startBackup() { ref.watch(errorBackupListProvider.notifier).empty(); ref.watch(backupProvider.notifier).startBackupProcess(); } return Scaffold( appBar: AppBar( elevation: 0, title: const Text( "Backup", style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold), ), leading: IconButton( onPressed: () { ref.watch(websocketProvider.notifier).listenUploadEvent(); AutoRouter.of(context).pop(true); }, splashRadius: 24, icon: const Icon( Icons.arrow_back_ios_rounded, )), ), body: Padding( padding: const EdgeInsets.only(left: 16.0, right: 16, bottom: 32), child: ListView( // crossAxisAlignment: CrossAxisAlignment.start, children: [ const Padding( padding: EdgeInsets.all(8.0), child: Text( "Backup Information", style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16), ), ), _buildFolderSelectionTile(), BackupInfoCard( title: "Total", subtitle: "All unique photos and videos from selected albums", info: "${backupState.allUniqueAssets.length}", ), BackupInfoCard( title: "Backup", subtitle: "Backed up photos and videos", info: "${backupState.selectedAlbumsBackupAssetsIds.length}", ), BackupInfoCard( title: "Remainder", subtitle: "Remaining photos and albums to back up from selection", info: "${backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length}", ), const Divider(), _buildBackupController(), const Divider(), _buildStorageInformation(), const Divider(), _buildCurrentBackupAssetInfoCard(), Padding( padding: const EdgeInsets.only( top: 24, ), child: Container( child: backupState.backupProgress == BackUpProgressEnum.inProgress ? ElevatedButton( style: ElevatedButton.styleFrom( primary: Colors.red[300], onPrimary: Colors.grey[50], padding: const EdgeInsets.all(14), ), onPressed: () { ref.read(backupProvider.notifier).cancelBackup(); }, child: const Text( "CANCEL", style: TextStyle( fontSize: 14, fontWeight: FontWeight.bold, ), ), ) : ElevatedButton( style: ElevatedButton.styleFrom( primary: Theme.of(context).primaryColor, onPrimary: Colors.grey[50], padding: const EdgeInsets.all(14), ), onPressed: shouldBackup ? startBackup : null, child: const Text( "START BACKUP", style: TextStyle( fontSize: 14, fontWeight: FontWeight.bold, ), ), ), ), ) ], ), ), ); } }