import 'package:auto_route/auto_route.dart'; import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; import 'package:fluttertoast/fluttertoast.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:immich_mobile/extensions/build_context_extensions.dart'; import 'package:immich_mobile/models/backup/available_album.model.dart'; import 'package:immich_mobile/providers/backup/backup.provider.dart'; import 'package:immich_mobile/routing/router.dart'; import 'package:immich_mobile/providers/haptic_feedback.provider.dart'; import 'package:immich_mobile/shared/ui/immich_toast.dart'; class AlbumInfoCard extends HookConsumerWidget { final AvailableAlbum album; const AlbumInfoCard({super.key, required this.album}); @override Widget build(BuildContext context, WidgetRef ref) { final bool isSelected = ref.watch(backupProvider).selectedBackupAlbums.contains(album); final bool isExcluded = ref.watch(backupProvider).excludedBackupAlbums.contains(album); final isDarkTheme = context.isDarkTheme; ColorFilter selectedFilter = ColorFilter.mode( context.primaryColor.withAlpha(100), BlendMode.darken, ); ColorFilter excludedFilter = ColorFilter.mode(Colors.red.withAlpha(75), BlendMode.darken); ColorFilter unselectedFilter = const ColorFilter.mode(Colors.black, BlendMode.color); buildSelectedTextBox() { if (isSelected) { return Chip( visualDensity: VisualDensity.compact, shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(5)), label: Text( "album_info_card_backup_album_included", style: TextStyle( fontSize: 10, color: isDarkTheme ? Colors.black : Colors.white, fontWeight: FontWeight.bold, ), ).tr(), backgroundColor: context.primaryColor, ); } else if (isExcluded) { return Chip( visualDensity: VisualDensity.compact, shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(5)), label: Text( "album_info_card_backup_album_excluded", style: TextStyle( fontSize: 10, color: isDarkTheme ? Colors.black : Colors.white, fontWeight: FontWeight.bold, ), ).tr(), backgroundColor: Colors.red[300], ); } return const SizedBox(); } buildImageFilter() { if (isSelected) { return selectedFilter; } else if (isExcluded) { return excludedFilter; } else { return unselectedFilter; } } return GestureDetector( onTap: () { ref.read(hapticFeedbackProvider.notifier).selectionClick(); if (isSelected) { ref.read(backupProvider.notifier).removeAlbumForBackup(album); } else { ref.read(backupProvider.notifier).addAlbumForBackup(album); } }, onDoubleTap: () { ref.read(hapticFeedbackProvider.notifier).selectionClick(); if (isExcluded) { // Remove from exclude album list ref.read(backupProvider.notifier).removeExcludedAlbumForBackup(album); } else { // Add to exclude album list if (album.id == 'isAll' || album.name == 'Recents') { ImmichToast.show( context: context, msg: 'Cannot exclude album contains all assets', toastType: ToastType.error, gravity: ToastGravity.BOTTOM, ); return; } ref.read(backupProvider.notifier).addExcludedAlbumForBackup(album); } }, child: Card( clipBehavior: Clip.hardEdge, margin: const EdgeInsets.all(1), shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(12), // if you need this side: BorderSide( color: isDarkTheme ? const Color.fromARGB(255, 37, 35, 35) : const Color(0xFFC9C9C9), width: 1, ), ), elevation: 0, borderOnForeground: false, child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ Expanded( child: Stack( clipBehavior: Clip.hardEdge, children: [ ColorFiltered( colorFilter: buildImageFilter(), child: const Image( width: double.infinity, height: double.infinity, image: AssetImage( 'assets/immich-logo.png', ), fit: BoxFit.cover, ), ), Positioned( bottom: 10, right: 25, child: buildSelectedTextBox(), ), ], ), ), Padding( padding: const EdgeInsets.only( left: 25, ), child: Row( crossAxisAlignment: CrossAxisAlignment.center, children: [ Expanded( child: Column( crossAxisAlignment: CrossAxisAlignment.start, mainAxisAlignment: MainAxisAlignment.center, children: [ Text( album.name, style: TextStyle( fontSize: 14, color: context.primaryColor, fontWeight: FontWeight.bold, ), ), Padding( padding: const EdgeInsets.only(top: 2.0), child: FutureBuilder( builder: ((context, snapshot) { if (snapshot.hasData) { return Text( snapshot.data.toString() + (album.isAll ? " (${'backup_all'.tr()})" : ""), style: TextStyle( fontSize: 12, color: Colors.grey[600], ), ); } return const Text("0"); }), future: album.assetCount, ), ), ], ), ), IconButton( onPressed: () { context.pushRoute( AlbumPreviewRoute(album: album.albumEntity), ); }, icon: Icon( Icons.image_outlined, color: context.primaryColor, size: 24, ), splashRadius: 25, ), ], ), ), ], ), ), ); } }