import 'dart:io'; import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; import 'package:flutter_map/flutter_map.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:immich_mobile/modules/asset_viewer/ui/description_input.dart'; import 'package:immich_mobile/modules/map/ui/map_thumbnail.dart'; import 'package:immich_mobile/shared/models/asset.dart'; import 'package:immich_mobile/shared/ui/drag_sheet.dart'; import 'package:latlong2/latlong.dart'; import 'package:immich_mobile/utils/bytes_units.dart'; import 'package:url_launcher/url_launcher.dart'; class ExifBottomSheet extends HookConsumerWidget { final Asset asset; const ExifBottomSheet({Key? key, required this.asset}) : super(key: key); bool get hasCoordinates => asset.exifInfo?.latitude != null && asset.exifInfo?.longitude != null && asset.exifInfo!.latitude! != 0 && asset.exifInfo!.longitude! != 0; String get formattedDateTime { final fileCreatedAt = asset.fileCreatedAt.toLocal(); final date = DateFormat.yMMMEd().format(fileCreatedAt); final time = DateFormat.jm().format(fileCreatedAt); return '$date • $time'; } Future _createCoordinatesUri() async { if (!hasCoordinates) { return null; } double latitude = asset.exifInfo!.latitude!; double longitude = asset.exifInfo!.longitude!; const zoomLevel = 16; if (Platform.isAndroid) { Uri uri = Uri( scheme: 'geo', host: '$latitude,$longitude', queryParameters: { 'z': '$zoomLevel', 'q': '$latitude,$longitude($formattedDateTime)', }, ); if (await canLaunchUrl(uri)) { return uri; } } else if (Platform.isIOS) { var params = { 'll': '$latitude,$longitude', 'q': formattedDateTime, 'z': '$zoomLevel', }; Uri uri = Uri.https('maps.apple.com', '/', params); if (await canLaunchUrl(uri)) { return uri; } } return Uri( scheme: 'https', host: 'openstreetmap.org', queryParameters: {'mlat': '$latitude', 'mlon': '$longitude'}, fragment: 'map=$zoomLevel/$latitude/$longitude', ); } @override Widget build(BuildContext context, WidgetRef ref) { final exifInfo = asset.exifInfo; var isDarkTheme = Theme.of(context).brightness == Brightness.dark; var textColor = isDarkTheme ? Colors.white : Colors.black; buildMap() { return Padding( padding: const EdgeInsets.symmetric(vertical: 16.0), child: LayoutBuilder( builder: (context, constraints) { return MapThumbnail( coords: LatLng( exifInfo?.latitude ?? 0, exifInfo?.longitude ?? 0, ), height: 150, zoom: 16.0, markers: [ Marker( anchorPos: AnchorPos.align(AnchorAlign.top), point: LatLng( exifInfo?.latitude ?? 0, exifInfo?.longitude ?? 0, ), builder: (ctx) => const Image( image: AssetImage('assets/location-pin.png'), ), ), ], onTap: (tapPosition, latLong) async { Uri? uri = await _createCoordinatesUri(); if (uri == null) { return; } debugPrint('Opening Map Uri: $uri'); launchUrl(uri); }, ); }, ), ); } buildSizeText(Asset a) { String resolution = a.width != null && a.height != null ? "${a.height} x ${a.width} " : ""; String fileSize = a.exifInfo?.fileSize != null ? formatBytes(a.exifInfo!.fileSize!) : ""; String text = resolution + fileSize; return text.isNotEmpty ? text : null; } buildDragHeader() { return const Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ SizedBox(height: 12), Align( alignment: Alignment.center, child: CustomDraggingHandle(), ), SizedBox(height: 12), ], ); } buildLocation() { // Guard no lat/lng if (!hasCoordinates) { return Container(); } return Column( children: [ // Location Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( "exif_bottom_sheet_location", style: TextStyle( fontSize: 11, color: textColor, fontWeight: FontWeight.bold, ), ).tr(), buildMap(), RichText( text: TextSpan( style: TextStyle( fontSize: 12, fontWeight: FontWeight.bold, color: textColor, fontFamily: 'WorkSans', ), children: [ if (exifInfo != null && exifInfo.city != null) TextSpan( text: exifInfo.city, ), if (exifInfo != null && exifInfo.city != null && exifInfo.state != null) const TextSpan( text: ", ", ), if (exifInfo != null && exifInfo.state != null) TextSpan( text: "${exifInfo.state}", ), ], ), ), Text( "${exifInfo!.latitude!.toStringAsFixed(4)}, ${exifInfo.longitude!.toStringAsFixed(4)}", style: const TextStyle(fontSize: 12), ), ], ), ], ); } buildDate() { return Text( formattedDateTime, style: const TextStyle( fontWeight: FontWeight.bold, fontSize: 14, ), ); } buildImageProperties() { // Helper to create the ListTile and avoid repeating code createImagePropertiesListStyle(title, subtitle) => ListTile( contentPadding: const EdgeInsets.all(0), dense: true, leading: Icon( Icons.image, color: textColor.withAlpha(200), ), titleAlignment: ListTileTitleAlignment.center, title: Text( title, style: TextStyle( fontWeight: FontWeight.bold, color: textColor, ), ), subtitle: subtitle, ); final imgSizeString = buildSizeText(asset); if (imgSizeString == null && asset.fileName.isNotEmpty) { // There is only filename return createImagePropertiesListStyle( asset.fileName, null, ); } else if (imgSizeString != null && asset.fileName.isNotEmpty) { // There is both filename and size information return createImagePropertiesListStyle( asset.fileName, Text(imgSizeString), ); } else if (imgSizeString != null && asset.fileName.isEmpty) { // There is only size information return createImagePropertiesListStyle( imgSizeString, null, ); } } buildDetail() { final imgProperties = buildImageProperties(); // There are no details if (imgProperties == null && (exifInfo == null || exifInfo.make == null)) { return Container(); } return Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Padding( padding: const EdgeInsets.only(bottom: 8.0), child: Text( "exif_bottom_sheet_details", style: TextStyle( fontSize: 11, color: textColor, fontWeight: FontWeight.bold, ), ).tr(), ), if (imgProperties != null) imgProperties, if (exifInfo?.make != null) ListTile( contentPadding: const EdgeInsets.all(0), dense: true, leading: Icon( Icons.camera, color: textColor.withAlpha(200), ), title: Text( "${exifInfo!.make} ${exifInfo.model}", style: TextStyle( color: textColor, fontWeight: FontWeight.bold, ), ), subtitle: Text( "ƒ/${exifInfo.fNumber} ${exifInfo.exposureTime} ${exifInfo.focalLength} mm ISO ${exifInfo.iso ?? ''} ", ), ), ], ); } return GestureDetector( onTap: () { // FocusScope.of(context).unfocus(); }, child: SingleChildScrollView( child: Card( shape: const RoundedRectangleBorder( borderRadius: BorderRadius.only( topLeft: Radius.circular(15), topRight: Radius.circular(15), ), ), margin: const EdgeInsets.all(0), child: Container( margin: const EdgeInsets.symmetric(horizontal: 16.0), child: LayoutBuilder( builder: (context, constraints) { if (constraints.maxWidth > 600) { // Two column return Padding( padding: const EdgeInsets.symmetric(horizontal: 12.0), child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ buildDragHeader(), buildDate(), if (asset.isRemote) DescriptionInput(asset: asset), Row( mainAxisAlignment: MainAxisAlignment.spaceEvenly, crossAxisAlignment: CrossAxisAlignment.start, children: [ Flexible( flex: hasCoordinates ? 5 : 0, child: Padding( padding: const EdgeInsets.only(right: 8.0), child: buildLocation(), ), ), Flexible( flex: 5, child: Padding( padding: const EdgeInsets.only(left: 8.0), child: buildDetail(), ), ), ], ), const SizedBox(height: 50), ], ), ); } // One column return Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ buildDragHeader(), buildDate(), if (asset.isRemote) DescriptionInput(asset: asset), const SizedBox(height: 8.0), buildLocation(), SizedBox(height: hasCoordinates ? 16.0 : 0.0), buildDetail(), const SizedBox(height: 50), ], ); }, ), ), ), ), ); } }