mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-25 07:49:05 -04:00 
			
		
		
		
	* Avoid creating unnecessary widgets * more flexible null handling and runtime errors prevention
		
			
				
	
	
		
			187 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			187 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'package:auto_route/auto_route.dart';
 | |
| import 'package:cached_network_image/cached_network_image.dart';
 | |
| import 'package:flutter/material.dart';
 | |
| import 'package:flutter_hooks/flutter_hooks.dart';
 | |
| import 'package:hive_flutter/hive_flutter.dart';
 | |
| import 'package:hooks_riverpod/hooks_riverpod.dart';
 | |
| import 'package:immich_mobile/constants/hive_box.dart';
 | |
| import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
 | |
| import 'package:immich_mobile/modules/sharing/providers/asset_selection.provider.dart';
 | |
| import 'package:immich_mobile/shared/models/immich_asset.model.dart';
 | |
| import 'package:immich_mobile/routing/router.dart';
 | |
| 
 | |
| class AlbumViewerThumbnail extends HookConsumerWidget {
 | |
|   final ImmichAsset asset;
 | |
| 
 | |
|   const AlbumViewerThumbnail({Key? key, required this.asset}) : super(key: key);
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context, WidgetRef ref) {
 | |
|     final cacheKey = useState(1);
 | |
|     var box = Hive.box(userInfoBox);
 | |
|     var thumbnailRequestUrl =
 | |
|         '${box.get(serverEndpointKey)}/asset/file?aid=${asset.deviceAssetId}&did=${asset.deviceId}&isThumb=true';
 | |
|     var deviceId = ref.watch(authenticationProvider).deviceId;
 | |
|     final selectedAssetsInAlbumViewer =
 | |
|         ref.watch(assetSelectionProvider).selectedAssetsInAlbumViewer;
 | |
|     final isMultiSelectionEnable =
 | |
|         ref.watch(assetSelectionProvider).isMultiselectEnable;
 | |
| 
 | |
|     _viewAsset() {
 | |
|       if (asset.type == 'IMAGE') {
 | |
|         AutoRouter.of(context).push(
 | |
|           ImageViewerRoute(
 | |
|             imageUrl:
 | |
|                 '${box.get(serverEndpointKey)}/asset/file?aid=${asset.deviceAssetId}&did=${asset.deviceId}&isThumb=false',
 | |
|             heroTag: asset.id,
 | |
|             thumbnailUrl: thumbnailRequestUrl,
 | |
|             asset: asset,
 | |
|           ),
 | |
|         );
 | |
|       } else {
 | |
|         AutoRouter.of(context).push(
 | |
|           VideoViewerRoute(
 | |
|               videoUrl:
 | |
|                   '${box.get(serverEndpointKey)}/asset/file?aid=${asset.deviceAssetId}&did=${asset.deviceId}',
 | |
|               asset: asset),
 | |
|         );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     BoxBorder drawBorderColor() {
 | |
|       if (selectedAssetsInAlbumViewer.contains(asset)) {
 | |
|         return Border.all(
 | |
|           color: Theme.of(context).primaryColorLight,
 | |
|           width: 10,
 | |
|         );
 | |
|       } else {
 | |
|         return const Border();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     _enableMultiSelection() {
 | |
|       ref.watch(assetSelectionProvider.notifier).enableMultiselection();
 | |
|       ref
 | |
|           .watch(assetSelectionProvider.notifier)
 | |
|           .addAssetsInAlbumViewer([asset]);
 | |
|     }
 | |
| 
 | |
|     _disableMultiSelection() {
 | |
|       ref.watch(assetSelectionProvider.notifier).disableMultiselection();
 | |
|     }
 | |
| 
 | |
|     _buildVideoLabel() {
 | |
|       return Positioned(
 | |
|         top: 5,
 | |
|         right: 5,
 | |
|         child: Row(
 | |
|           children: [
 | |
|             Text(
 | |
|               asset.duration.toString().substring(0, 7),
 | |
|               style: const TextStyle(
 | |
|                 color: Colors.white,
 | |
|                 fontSize: 10,
 | |
|               ),
 | |
|             ),
 | |
|             const Icon(
 | |
|               Icons.play_circle_outline_rounded,
 | |
|               color: Colors.white,
 | |
|             ),
 | |
|           ],
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     _buildAssetStoreLocationIcon() {
 | |
|       return Positioned(
 | |
|         right: 10,
 | |
|         bottom: 5,
 | |
|         child: Icon(
 | |
|           (deviceId != asset.deviceId)
 | |
|               ? Icons.cloud_done_outlined
 | |
|               : Icons.photo_library_rounded,
 | |
|           color: Colors.white,
 | |
|           size: 18,
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     _buildAssetSelectionIcon() {
 | |
|       bool isSelected = selectedAssetsInAlbumViewer.contains(asset);
 | |
| 
 | |
|       return Positioned(
 | |
|         left: 10,
 | |
|         top: 5,
 | |
|         child: isSelected
 | |
|             ? Icon(
 | |
|                 Icons.check_circle_rounded,
 | |
|                 color: Theme.of(context).primaryColor,
 | |
|               )
 | |
|             : const Icon(
 | |
|                 Icons.check_circle_outline_rounded,
 | |
|                 color: Colors.white,
 | |
|               ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     _buildThumbnailImage() {
 | |
|       return Container(
 | |
|         decoration: BoxDecoration(border: drawBorderColor()),
 | |
|         child: CachedNetworkImage(
 | |
|           cacheKey: "${asset.id}-${cacheKey.value}",
 | |
|           width: 300,
 | |
|           height: 300,
 | |
|           memCacheHeight: 200,
 | |
|           fit: BoxFit.cover,
 | |
|           imageUrl: thumbnailRequestUrl,
 | |
|           httpHeaders: {"Authorization": "Bearer ${box.get(accessTokenKey)}"},
 | |
|           fadeInDuration: const Duration(milliseconds: 250),
 | |
|           progressIndicatorBuilder: (context, url, downloadProgress) =>
 | |
|               Transform.scale(
 | |
|             scale: 0.2,
 | |
|             child: CircularProgressIndicator(value: downloadProgress.progress),
 | |
|           ),
 | |
|           errorWidget: (context, url, error) {
 | |
|             return Icon(
 | |
|               Icons.image_not_supported_outlined,
 | |
|               color: Theme.of(context).primaryColor,
 | |
|             );
 | |
|           },
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     _handleSelectionGesture() {
 | |
|       if (selectedAssetsInAlbumViewer.contains(asset)) {
 | |
|         ref
 | |
|             .watch(assetSelectionProvider.notifier)
 | |
|             .removeAssetsInAlbumViewer([asset]);
 | |
| 
 | |
|         if (selectedAssetsInAlbumViewer.isEmpty) {
 | |
|           _disableMultiSelection();
 | |
|         }
 | |
|       } else {
 | |
|         ref
 | |
|             .watch(assetSelectionProvider.notifier)
 | |
|             .addAssetsInAlbumViewer([asset]);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return GestureDetector(
 | |
|       onTap: isMultiSelectionEnable ? _handleSelectionGesture : _viewAsset,
 | |
|       onLongPress: _enableMultiSelection,
 | |
|       child: Hero(
 | |
|         tag: asset.id,
 | |
|         child: Stack(
 | |
|           children: [
 | |
|             _buildThumbnailImage(),
 | |
|             _buildAssetStoreLocationIcon(),
 | |
|             if (asset.type != 'IMAGE') _buildVideoLabel(),
 | |
|             if (isMultiSelectionEnable) _buildAssetSelectionIcon(),
 | |
|           ],
 | |
|         ),
 | |
|       ),
 | |
|     );
 | |
|   }
 | |
| }
 |