mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-26 00:02:34 -04:00 
			
		
		
		
	* update dependencies * update flutter version reference * update flutter version reference * update AndroidManifest with flutter_web_auth_2 * chore: lock file flutter version * fix: ios build
		
			
				
	
	
		
			326 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			326 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:math' as math;
 | |
| import 'package:auto_route/auto_route.dart';
 | |
| import 'package:easy_localization/easy_localization.dart';
 | |
| import 'package:flutter/material.dart';
 | |
| import 'package:flutter/services.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/widgets/search/thumbnail_with_info.dart';
 | |
| import 'package:immich_mobile/models/shared_link/shared_link.model.dart';
 | |
| import 'package:immich_mobile/providers/shared_link.provider.dart';
 | |
| import 'package:immich_mobile/routing/router.dart';
 | |
| import 'package:immich_mobile/providers/server_info.provider.dart';
 | |
| import 'package:immich_mobile/widgets/common/confirm_dialog.dart';
 | |
| import 'package:immich_mobile/widgets/common/immich_toast.dart';
 | |
| import 'package:immich_mobile/utils/image_url_builder.dart';
 | |
| import 'package:immich_mobile/utils/url_helper.dart';
 | |
| 
 | |
| class SharedLinkItem extends ConsumerWidget {
 | |
|   final SharedLink sharedLink;
 | |
| 
 | |
|   const SharedLinkItem(this.sharedLink, {super.key});
 | |
| 
 | |
|   bool isExpired() {
 | |
|     if (sharedLink.expiresAt != null) {
 | |
|       return DateTime.now().isAfter(sharedLink.expiresAt!);
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   Widget getExpiryDuration(bool isDarkMode) {
 | |
|     var expiresText = "shared_link_expires_never".tr();
 | |
|     if (sharedLink.expiresAt != null) {
 | |
|       if (isExpired()) {
 | |
|         return Text(
 | |
|           "shared_link_expired",
 | |
|           style: TextStyle(color: Colors.red[300]),
 | |
|         ).tr();
 | |
|       }
 | |
|       final difference = sharedLink.expiresAt!.difference(DateTime.now());
 | |
|       debugPrint("Difference: $difference");
 | |
|       if (difference.inDays > 0) {
 | |
|         var dayDifference = difference.inDays;
 | |
|         if (difference.inHours % 24 > 12) {
 | |
|           dayDifference += 1;
 | |
|         }
 | |
|         expiresText =
 | |
|             "shared_link_expires_days".tr(args: [dayDifference.toString()]);
 | |
|       } else if (difference.inHours > 0) {
 | |
|         expiresText = "shared_link_expires_hours"
 | |
|             .tr(args: [difference.inHours.toString()]);
 | |
|       } else if (difference.inMinutes > 0) {
 | |
|         expiresText = "shared_link_expires_minutes"
 | |
|             .tr(args: [difference.inMinutes.toString()]);
 | |
|       } else if (difference.inSeconds > 0) {
 | |
|         expiresText = "shared_link_expires_seconds"
 | |
|             .tr(args: [difference.inSeconds.toString()]);
 | |
|       }
 | |
|     }
 | |
|     return Text(
 | |
|       expiresText,
 | |
|       style: TextStyle(color: isDarkMode ? Colors.grey[400] : Colors.grey[600]),
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context, WidgetRef ref) {
 | |
|     final colorScheme = context.colorScheme;
 | |
|     final isDarkMode = colorScheme.brightness == Brightness.dark;
 | |
|     final thumbnailUrl = sharedLink.thumbAssetId != null
 | |
|         ? getThumbnailUrlForRemoteId(sharedLink.thumbAssetId!)
 | |
|         : null;
 | |
|     final imageSize = math.min(context.width / 4, 100.0);
 | |
| 
 | |
|     void copyShareLinkToClipboard() {
 | |
|       final externalDomain = ref.read(
 | |
|         serverInfoProvider.select((s) => s.serverConfig.externalDomain),
 | |
|       );
 | |
|       var serverUrl =
 | |
|           externalDomain.isNotEmpty ? externalDomain : getServerUrl();
 | |
|       if (serverUrl != null && !serverUrl.endsWith('/')) {
 | |
|         serverUrl += '/';
 | |
|       }
 | |
|       if (serverUrl == null) {
 | |
|         ImmichToast.show(
 | |
|           context: context,
 | |
|           gravity: ToastGravity.BOTTOM,
 | |
|           toastType: ToastType.error,
 | |
|           msg: "shared_link_error_server_url_fetch".tr(),
 | |
|         );
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       Clipboard.setData(
 | |
|         ClipboardData(text: "${serverUrl}share/${sharedLink.key}"),
 | |
|       ).then((_) {
 | |
|         context.scaffoldMessenger.showSnackBar(
 | |
|           SnackBar(
 | |
|             content: Text(
 | |
|               "shared_link_clipboard_copied_massage",
 | |
|               style: context.textTheme.bodyLarge?.copyWith(
 | |
|                 color: context.primaryColor,
 | |
|               ),
 | |
|             ).tr(),
 | |
|             duration: const Duration(seconds: 2),
 | |
|           ),
 | |
|         );
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     Future<void> deleteShareLink() async {
 | |
|       return showDialog(
 | |
|         context: context,
 | |
|         builder: (BuildContext context) {
 | |
|           return ConfirmDialog(
 | |
|             title: "delete_shared_link_dialog_title",
 | |
|             content: "delete_shared_link_dialog_content",
 | |
|             onOk: () => ref
 | |
|                 .read(sharedLinksStateProvider.notifier)
 | |
|                 .deleteLink(sharedLink.id),
 | |
|           );
 | |
|         },
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     Widget buildThumbnail() {
 | |
|       if (thumbnailUrl == null) {
 | |
|         return Container(
 | |
|           height: imageSize * 1.2,
 | |
|           width: imageSize,
 | |
|           decoration: BoxDecoration(
 | |
|             color: isDarkMode ? Colors.grey[800] : Colors.grey[200],
 | |
|           ),
 | |
|           child: Center(
 | |
|             child: Icon(
 | |
|               Icons.image_not_supported_outlined,
 | |
|               color: isDarkMode ? Colors.grey[100] : Colors.grey[700],
 | |
|             ),
 | |
|           ),
 | |
|         );
 | |
|       }
 | |
|       return SizedBox(
 | |
|         height: imageSize * 1.2,
 | |
|         width: imageSize,
 | |
|         child: Padding(
 | |
|           padding: const EdgeInsets.only(right: 4.0),
 | |
|           child: ThumbnailWithInfo(
 | |
|             imageUrl: thumbnailUrl,
 | |
|             key: key,
 | |
|             textInfo: '',
 | |
|             noImageIcon: Icons.image_not_supported_outlined,
 | |
|             onTap: () {},
 | |
|           ),
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     Widget buildInfoChip(String labelText) {
 | |
|       return Padding(
 | |
|         padding: const EdgeInsets.only(right: 10),
 | |
|         child: Chip(
 | |
|           backgroundColor: colorScheme.primary,
 | |
|           label: Text(
 | |
|             labelText,
 | |
|             style: TextStyle(
 | |
|               fontSize: 11,
 | |
|               fontWeight: FontWeight.w500,
 | |
|               color: isDarkMode ? Colors.black : Colors.white,
 | |
|             ),
 | |
|           ),
 | |
|           shape: const RoundedRectangleBorder(
 | |
|             borderRadius: BorderRadius.all(Radius.circular(25)),
 | |
|           ),
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     Widget buildBottomInfo() {
 | |
|       return Row(
 | |
|         children: [
 | |
|           if (sharedLink.allowUpload)
 | |
|             buildInfoChip("shared_link_info_chip_upload".tr()),
 | |
|           if (sharedLink.allowDownload)
 | |
|             buildInfoChip("shared_link_info_chip_download".tr()),
 | |
|           if (sharedLink.showMetadata)
 | |
|             buildInfoChip("shared_link_info_chip_metadata".tr()),
 | |
|         ],
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     Widget buildSharedLinkActions() {
 | |
|       const actionIconSize = 20.0;
 | |
|       return Row(
 | |
|         children: [
 | |
|           IconButton(
 | |
|             splashRadius: 25,
 | |
|             constraints: const BoxConstraints(),
 | |
|             iconSize: actionIconSize,
 | |
|             icon: const Icon(Icons.delete_outline),
 | |
|             style: const ButtonStyle(
 | |
|               tapTargetSize:
 | |
|                   MaterialTapTargetSize.shrinkWrap, // the '2023' part
 | |
|             ),
 | |
|             onPressed: deleteShareLink,
 | |
|           ),
 | |
|           IconButton(
 | |
|             splashRadius: 25,
 | |
|             constraints: const BoxConstraints(),
 | |
|             iconSize: actionIconSize,
 | |
|             icon: const Icon(Icons.edit_outlined),
 | |
|             style: const ButtonStyle(
 | |
|               tapTargetSize:
 | |
|                   MaterialTapTargetSize.shrinkWrap, // the '2023' part
 | |
|             ),
 | |
|             onPressed: () => context
 | |
|                 .pushRoute(SharedLinkEditRoute(existingLink: sharedLink)),
 | |
|           ),
 | |
|           IconButton(
 | |
|             splashRadius: 25,
 | |
|             constraints: const BoxConstraints(),
 | |
|             iconSize: actionIconSize,
 | |
|             icon: const Icon(Icons.copy_outlined),
 | |
|             style: const ButtonStyle(
 | |
|               tapTargetSize:
 | |
|                   MaterialTapTargetSize.shrinkWrap, // the '2023' part
 | |
|             ),
 | |
|             onPressed: copyShareLinkToClipboard,
 | |
|           ),
 | |
|         ],
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     Widget buildSharedLinkDetails() {
 | |
|       return Column(
 | |
|         crossAxisAlignment: CrossAxisAlignment.stretch,
 | |
|         children: [
 | |
|           getExpiryDuration(isDarkMode),
 | |
|           Padding(
 | |
|             padding: const EdgeInsets.only(top: 5),
 | |
|             child: Tooltip(
 | |
|               verticalOffset: 0,
 | |
|               decoration: BoxDecoration(
 | |
|                 color: colorScheme.primary.withValues(alpha: 0.9),
 | |
|                 borderRadius: BorderRadius.circular(10),
 | |
|               ),
 | |
|               textStyle: TextStyle(
 | |
|                 color: isDarkMode ? Colors.black : Colors.white,
 | |
|                 fontWeight: FontWeight.bold,
 | |
|               ),
 | |
|               message: sharedLink.title,
 | |
|               preferBelow: false,
 | |
|               triggerMode: TooltipTriggerMode.tap,
 | |
|               child: Text(
 | |
|                 sharedLink.title,
 | |
|                 style: TextStyle(
 | |
|                   color: colorScheme.primary,
 | |
|                   fontWeight: FontWeight.bold,
 | |
|                   overflow: TextOverflow.ellipsis,
 | |
|                 ),
 | |
|               ),
 | |
|             ),
 | |
|           ),
 | |
|           Row(
 | |
|             mainAxisSize: MainAxisSize.max,
 | |
|             mainAxisAlignment: MainAxisAlignment.spaceBetween,
 | |
|             children: [
 | |
|               Expanded(
 | |
|                 child: Tooltip(
 | |
|                   verticalOffset: 0,
 | |
|                   decoration: BoxDecoration(
 | |
|                     color: colorScheme.primary.withValues(alpha: 0.9),
 | |
|                     borderRadius: BorderRadius.circular(10),
 | |
|                   ),
 | |
|                   textStyle: TextStyle(
 | |
|                     color: isDarkMode ? Colors.black : Colors.white,
 | |
|                     fontWeight: FontWeight.bold,
 | |
|                   ),
 | |
|                   message: sharedLink.description ?? "",
 | |
|                   preferBelow: false,
 | |
|                   triggerMode: TooltipTriggerMode.tap,
 | |
|                   child: Text(
 | |
|                     sharedLink.description ?? "",
 | |
|                     overflow: TextOverflow.ellipsis,
 | |
|                   ),
 | |
|                 ),
 | |
|               ),
 | |
|               Padding(
 | |
|                 padding: const EdgeInsets.only(right: 15),
 | |
|                 child: buildSharedLinkActions(),
 | |
|               ),
 | |
|             ],
 | |
|           ),
 | |
|           buildBottomInfo(),
 | |
|         ],
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     return Column(
 | |
|       crossAxisAlignment: CrossAxisAlignment.start,
 | |
|       children: [
 | |
|         Row(
 | |
|           crossAxisAlignment: CrossAxisAlignment.start,
 | |
|           children: [
 | |
|             Padding(
 | |
|               padding: const EdgeInsets.only(left: 15),
 | |
|               child: buildThumbnail(),
 | |
|             ),
 | |
|             Expanded(
 | |
|               child: Padding(
 | |
|                 padding: const EdgeInsets.only(left: 15),
 | |
|                 child: buildSharedLinkDetails(),
 | |
|               ),
 | |
|             ),
 | |
|           ],
 | |
|         ),
 | |
|         const Padding(
 | |
|           padding: EdgeInsets.all(20),
 | |
|           child: Divider(
 | |
|             height: 0,
 | |
|           ),
 | |
|         ),
 | |
|       ],
 | |
|     );
 | |
|   }
 | |
| }
 |