mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-24 23:39:03 -04:00 
			
		
		
		
	* enable border radius, switch exp, const constructor * regenerate provider * more formatting --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
		
			
				
	
	
		
			300 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			300 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| 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/providers/album/album.provider.dart';
 | |
| import 'package:immich_mobile/providers/backup/backup.provider.dart';
 | |
| import 'package:immich_mobile/services/app_settings.service.dart';
 | |
| import 'package:immich_mobile/utils/hooks/app_settings_update_hook.dart';
 | |
| import 'package:immich_mobile/widgets/backup/album_info_card.dart';
 | |
| import 'package:immich_mobile/widgets/backup/album_info_list_tile.dart';
 | |
| import 'package:immich_mobile/widgets/settings/settings_switch_list_tile.dart';
 | |
| 
 | |
| @RoutePage()
 | |
| class BackupAlbumSelectionPage extends HookConsumerWidget {
 | |
|   const BackupAlbumSelectionPage({super.key});
 | |
|   @override
 | |
|   Widget build(BuildContext context, WidgetRef ref) {
 | |
|     final selectedBackupAlbums = ref.watch(backupProvider).selectedBackupAlbums;
 | |
|     final excludedBackupAlbums = ref.watch(backupProvider).excludedBackupAlbums;
 | |
|     final enableSyncUploadAlbum =
 | |
|         useAppSettingsState(AppSettingsEnum.syncAlbums);
 | |
|     final isDarkTheme = context.isDarkTheme;
 | |
|     final albums = ref.watch(backupProvider).availableAlbums;
 | |
| 
 | |
|     useEffect(
 | |
|       () {
 | |
|         ref.watch(backupProvider.notifier).getBackupInfo();
 | |
|         return null;
 | |
|       },
 | |
|       [],
 | |
|     );
 | |
| 
 | |
|     buildAlbumSelectionList() {
 | |
|       if (albums.isEmpty) {
 | |
|         return const SliverToBoxAdapter(
 | |
|           child: Center(
 | |
|             child: CircularProgressIndicator(),
 | |
|           ),
 | |
|         );
 | |
|       }
 | |
| 
 | |
|       return SliverPadding(
 | |
|         padding: const EdgeInsets.symmetric(vertical: 12.0),
 | |
|         sliver: SliverList(
 | |
|           delegate: SliverChildBuilderDelegate(
 | |
|             ((context, index) {
 | |
|               return AlbumInfoListTile(
 | |
|                 album: albums[index],
 | |
|               );
 | |
|             }),
 | |
|             childCount: albums.length,
 | |
|           ),
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     buildAlbumSelectionGrid() {
 | |
|       if (albums.isEmpty) {
 | |
|         return const SliverToBoxAdapter(
 | |
|           child: Center(
 | |
|             child: CircularProgressIndicator(),
 | |
|           ),
 | |
|         );
 | |
|       }
 | |
| 
 | |
|       return SliverPadding(
 | |
|         padding: const EdgeInsets.all(12.0),
 | |
|         sliver: SliverGrid.builder(
 | |
|           gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
 | |
|             maxCrossAxisExtent: 300,
 | |
|             mainAxisSpacing: 12,
 | |
|             crossAxisSpacing: 12,
 | |
|           ),
 | |
|           itemCount: albums.length,
 | |
|           itemBuilder: ((context, index) {
 | |
|             return AlbumInfoCard(
 | |
|               album: albums[index],
 | |
|             );
 | |
|           }),
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     buildSelectedAlbumNameChip() {
 | |
|       return selectedBackupAlbums.map((album) {
 | |
|         void removeSelection() =>
 | |
|             ref.read(backupProvider.notifier).removeAlbumForBackup(album);
 | |
| 
 | |
|         return Padding(
 | |
|           padding: const EdgeInsets.only(right: 8.0),
 | |
|           child: GestureDetector(
 | |
|             onTap: removeSelection,
 | |
|             child: Chip(
 | |
|               label: Text(
 | |
|                 album.name,
 | |
|                 style: TextStyle(
 | |
|                   fontSize: 12,
 | |
|                   color: isDarkTheme ? Colors.black : Colors.white,
 | |
|                   fontWeight: FontWeight.bold,
 | |
|                 ),
 | |
|               ),
 | |
|               backgroundColor: context.primaryColor,
 | |
|               deleteIconColor: isDarkTheme ? Colors.black : Colors.white,
 | |
|               deleteIcon: const Icon(
 | |
|                 Icons.cancel_rounded,
 | |
|                 size: 15,
 | |
|               ),
 | |
|               onDeleted: removeSelection,
 | |
|             ),
 | |
|           ),
 | |
|         );
 | |
|       }).toSet();
 | |
|     }
 | |
| 
 | |
|     buildExcludedAlbumNameChip() {
 | |
|       return excludedBackupAlbums.map((album) {
 | |
|         void removeSelection() {
 | |
|           ref
 | |
|               .watch(backupProvider.notifier)
 | |
|               .removeExcludedAlbumForBackup(album);
 | |
|         }
 | |
| 
 | |
|         return GestureDetector(
 | |
|           onTap: removeSelection,
 | |
|           child: Padding(
 | |
|             padding: const EdgeInsets.only(right: 8.0),
 | |
|             child: Chip(
 | |
|               label: Text(
 | |
|                 album.name,
 | |
|                 style: TextStyle(
 | |
|                   fontSize: 12,
 | |
|                   color: context.scaffoldBackgroundColor,
 | |
|                   fontWeight: FontWeight.bold,
 | |
|                 ),
 | |
|               ),
 | |
|               backgroundColor: Colors.red[300],
 | |
|               deleteIconColor: context.scaffoldBackgroundColor,
 | |
|               deleteIcon: const Icon(
 | |
|                 Icons.cancel_rounded,
 | |
|                 size: 15,
 | |
|               ),
 | |
|               onDeleted: removeSelection,
 | |
|             ),
 | |
|           ),
 | |
|         );
 | |
|       }).toSet();
 | |
|     }
 | |
| 
 | |
|     handleSyncAlbumToggle(bool isEnable) async {
 | |
|       if (isEnable) {
 | |
|         await ref.read(albumProvider.notifier).refreshRemoteAlbums();
 | |
|         for (final album in selectedBackupAlbums) {
 | |
|           await ref.read(albumProvider.notifier).createSyncAlbum(album.name);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return Scaffold(
 | |
|       appBar: AppBar(
 | |
|         leading: IconButton(
 | |
|           onPressed: () => context.maybePop(),
 | |
|           icon: const Icon(Icons.arrow_back_ios_rounded),
 | |
|         ),
 | |
|         title: const Text(
 | |
|           "backup_album_selection_page_select_albums",
 | |
|         ).tr(),
 | |
|         elevation: 0,
 | |
|       ),
 | |
|       body: CustomScrollView(
 | |
|         physics: const ClampingScrollPhysics(),
 | |
|         slivers: [
 | |
|           SliverToBoxAdapter(
 | |
|             child: Column(
 | |
|               crossAxisAlignment: CrossAxisAlignment.start,
 | |
|               children: [
 | |
|                 Padding(
 | |
|                   padding: const EdgeInsets.symmetric(
 | |
|                     vertical: 8.0,
 | |
|                     horizontal: 16.0,
 | |
|                   ),
 | |
|                   child: Text(
 | |
|                     "backup_album_selection_page_selection_info",
 | |
|                     style: context.textTheme.titleSmall,
 | |
|                   ).tr(),
 | |
|                 ),
 | |
|                 // Selected Album Chips
 | |
| 
 | |
|                 Padding(
 | |
|                   padding: const EdgeInsets.symmetric(horizontal: 16.0),
 | |
|                   child: Wrap(
 | |
|                     children: [
 | |
|                       ...buildSelectedAlbumNameChip(),
 | |
|                       ...buildExcludedAlbumNameChip(),
 | |
|                     ],
 | |
|                   ),
 | |
|                 ),
 | |
| 
 | |
|                 SettingsSwitchListTile(
 | |
|                   valueNotifier: enableSyncUploadAlbum,
 | |
|                   title: "sync_albums".tr(),
 | |
|                   subtitle: "sync_upload_album_setting_subtitle".tr(),
 | |
|                   contentPadding: const EdgeInsets.symmetric(horizontal: 16),
 | |
|                   titleStyle: context.textTheme.bodyLarge?.copyWith(
 | |
|                     fontWeight: FontWeight.bold,
 | |
|                   ),
 | |
|                   subtitleStyle: context.textTheme.labelLarge?.copyWith(
 | |
|                     color: context.colorScheme.primary,
 | |
|                   ),
 | |
|                   onChanged: handleSyncAlbumToggle,
 | |
|                 ),
 | |
| 
 | |
|                 ListTile(
 | |
|                   title: Text(
 | |
|                     "backup_album_selection_page_albums_device".tr(
 | |
|                       namedArgs: {
 | |
|                         'count': ref
 | |
|                             .watch(backupProvider)
 | |
|                             .availableAlbums
 | |
|                             .length
 | |
|                             .toString(),
 | |
|                       },
 | |
|                     ),
 | |
|                     style: context.textTheme.titleSmall,
 | |
|                   ),
 | |
|                   subtitle: Padding(
 | |
|                     padding: const EdgeInsets.symmetric(vertical: 8.0),
 | |
|                     child: Text(
 | |
|                       "backup_album_selection_page_albums_tap",
 | |
|                       style: context.textTheme.labelLarge?.copyWith(
 | |
|                         color: context.primaryColor,
 | |
|                       ),
 | |
|                     ).tr(),
 | |
|                   ),
 | |
|                   trailing: IconButton(
 | |
|                     splashRadius: 16,
 | |
|                     icon: Icon(
 | |
|                       Icons.info,
 | |
|                       size: 20,
 | |
|                       color: context.primaryColor,
 | |
|                     ),
 | |
|                     onPressed: () {
 | |
|                       // show the dialog
 | |
|                       showDialog(
 | |
|                         context: context,
 | |
|                         builder: (BuildContext context) {
 | |
|                           return AlertDialog(
 | |
|                             shape: const RoundedRectangleBorder(
 | |
|                               borderRadius: BorderRadius.all(
 | |
|                                 Radius.circular(10),
 | |
|                               ),
 | |
|                             ),
 | |
|                             elevation: 5,
 | |
|                             title: Text(
 | |
|                               'backup_album_selection_page_selection_info',
 | |
|                               style: TextStyle(
 | |
|                                 fontSize: 16,
 | |
|                                 fontWeight: FontWeight.bold,
 | |
|                                 color: context.primaryColor,
 | |
|                               ),
 | |
|                             ).tr(),
 | |
|                             content: SingleChildScrollView(
 | |
|                               child: ListBody(
 | |
|                                 children: [
 | |
|                                   const Text(
 | |
|                                     'backup_album_selection_page_assets_scatter',
 | |
|                                     style: TextStyle(
 | |
|                                       fontSize: 14,
 | |
|                                     ),
 | |
|                                   ).tr(),
 | |
|                                 ],
 | |
|                               ),
 | |
|                             ),
 | |
|                           );
 | |
|                         },
 | |
|                       );
 | |
|                     },
 | |
|                   ),
 | |
|                 ),
 | |
| 
 | |
|                 // buildSearchBar(),
 | |
|               ],
 | |
|             ),
 | |
|           ),
 | |
|           SliverLayoutBuilder(
 | |
|             builder: (context, constraints) {
 | |
|               if (constraints.crossAxisExtent > 600) {
 | |
|                 return buildAlbumSelectionGrid();
 | |
|               } else {
 | |
|                 return buildAlbumSelectionList();
 | |
|               }
 | |
|             },
 | |
|           ),
 | |
|         ],
 | |
|       ),
 | |
|     );
 | |
|   }
 | |
| }
 |