mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-31 10:37:11 -04:00 
			
		
		
		
	* fix: empty placeholders * fix: use namedArgs --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
		
			
				
	
	
		
			202 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			202 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:async';
 | |
| import 'dart:typed_data';
 | |
| import 'dart:ui';
 | |
| 
 | |
| 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/entities/asset.entity.dart';
 | |
| import 'package:immich_mobile/extensions/build_context_extensions.dart';
 | |
| import 'package:immich_mobile/providers/album/album.provider.dart';
 | |
| import 'package:immich_mobile/repositories/file_media.repository.dart';
 | |
| import 'package:immich_mobile/routing/router.dart';
 | |
| import 'package:immich_mobile/widgets/common/immich_toast.dart';
 | |
| import 'package:path/path.dart' as p;
 | |
| 
 | |
| /// A stateless widget that provides functionality for editing an image.
 | |
| ///
 | |
| /// This widget allows users to edit an image provided either as an [Asset] or
 | |
| /// directly as an [Image]. It ensures that exactly one of these is provided.
 | |
| ///
 | |
| /// It also includes a conversion method to convert an [Image] to a [Uint8List] to save the image on the user's phone
 | |
| /// They automatically navigate to the [HomePage] with the edited image saved and they eventually get backed up to the server.
 | |
| @immutable
 | |
| @RoutePage()
 | |
| class EditImagePage extends ConsumerWidget {
 | |
|   final Asset asset;
 | |
|   final Image image;
 | |
|   final bool isEdited;
 | |
| 
 | |
|   const EditImagePage({
 | |
|     super.key,
 | |
|     required this.asset,
 | |
|     required this.image,
 | |
|     required this.isEdited,
 | |
|   });
 | |
|   Future<Uint8List> _imageToUint8List(Image image) async {
 | |
|     final Completer<Uint8List> completer = Completer();
 | |
|     image.image.resolve(const ImageConfiguration()).addListener(
 | |
|           ImageStreamListener(
 | |
|             (ImageInfo info, bool _) {
 | |
|               info.image
 | |
|                   .toByteData(format: ImageByteFormat.png)
 | |
|                   .then((byteData) {
 | |
|                 if (byteData != null) {
 | |
|                   completer.complete(byteData.buffer.asUint8List());
 | |
|                 } else {
 | |
|                   completer.completeError('Failed to convert image to bytes');
 | |
|                 }
 | |
|               });
 | |
|             },
 | |
|             onError: (exception, stackTrace) =>
 | |
|                 completer.completeError(exception),
 | |
|           ),
 | |
|         );
 | |
|     return completer.future;
 | |
|   }
 | |
| 
 | |
|   Future<void> _saveEditedImage(
 | |
|     BuildContext context,
 | |
|     Asset asset,
 | |
|     Image image,
 | |
|     WidgetRef ref,
 | |
|   ) async {
 | |
|     try {
 | |
|       final Uint8List imageData = await _imageToUint8List(image);
 | |
|       await ref.read(fileMediaRepositoryProvider).saveImage(
 | |
|             imageData,
 | |
|             title: "${p.withoutExtension(asset.fileName)}_edited.jpg",
 | |
|           );
 | |
|       await ref.read(albumProvider.notifier).refreshDeviceAlbums();
 | |
|       context.navigator.popUntil((route) => route.isFirst);
 | |
|       ImmichToast.show(
 | |
|         durationInSecond: 3,
 | |
|         context: context,
 | |
|         msg: 'Image Saved!',
 | |
|         gravity: ToastGravity.CENTER,
 | |
|       );
 | |
|     } catch (e) {
 | |
|       ImmichToast.show(
 | |
|         durationInSecond: 6,
 | |
|         context: context,
 | |
|         msg: "error_saving_image".tr(namedArgs: {'error': e.toString()}),
 | |
|         gravity: ToastGravity.CENTER,
 | |
|       );
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context, WidgetRef ref) {
 | |
|     return Scaffold(
 | |
|       appBar: AppBar(
 | |
|         title: Text("edit".tr()),
 | |
|         backgroundColor: context.scaffoldBackgroundColor,
 | |
|         leading: IconButton(
 | |
|           icon: Icon(
 | |
|             Icons.close_rounded,
 | |
|             color: context.primaryColor,
 | |
|             size: 24,
 | |
|           ),
 | |
|           onPressed: () => context.navigator.popUntil((route) => route.isFirst),
 | |
|         ),
 | |
|         actions: <Widget>[
 | |
|           TextButton(
 | |
|             onPressed: isEdited
 | |
|                 ? () => _saveEditedImage(context, asset, image, ref)
 | |
|                 : null,
 | |
|             child: Text(
 | |
|               "save_to_gallery".tr(),
 | |
|               style: TextStyle(
 | |
|                 color: isEdited ? context.primaryColor : Colors.grey,
 | |
|               ),
 | |
|             ),
 | |
|           ),
 | |
|         ],
 | |
|       ),
 | |
|       backgroundColor: context.scaffoldBackgroundColor,
 | |
|       body: Center(
 | |
|         child: ConstrainedBox(
 | |
|           constraints: BoxConstraints(
 | |
|             maxHeight: context.height * 0.7,
 | |
|             maxWidth: context.width * 0.9,
 | |
|           ),
 | |
|           child: Container(
 | |
|             decoration: BoxDecoration(
 | |
|               borderRadius: BorderRadius.circular(7),
 | |
|               boxShadow: [
 | |
|                 BoxShadow(
 | |
|                   color: Colors.black.withValues(alpha: 0.2),
 | |
|                   spreadRadius: 2,
 | |
|                   blurRadius: 10,
 | |
|                   offset: const Offset(0, 3),
 | |
|                 ),
 | |
|               ],
 | |
|             ),
 | |
|             child: ClipRRect(
 | |
|               borderRadius: BorderRadius.circular(7),
 | |
|               child: Image(
 | |
|                 image: image.image,
 | |
|                 fit: BoxFit.contain,
 | |
|               ),
 | |
|             ),
 | |
|           ),
 | |
|         ),
 | |
|       ),
 | |
|       bottomNavigationBar: Container(
 | |
|         height: 70,
 | |
|         margin: const EdgeInsets.only(bottom: 60, right: 10, left: 10, top: 10),
 | |
|         decoration: BoxDecoration(
 | |
|           color: context.scaffoldBackgroundColor,
 | |
|           borderRadius: BorderRadius.circular(30),
 | |
|         ),
 | |
|         child: Row(
 | |
|           mainAxisAlignment: MainAxisAlignment.spaceEvenly,
 | |
|           children: <Widget>[
 | |
|             Column(
 | |
|               mainAxisAlignment: MainAxisAlignment.center,
 | |
|               children: <Widget>[
 | |
|                 IconButton(
 | |
|                   icon: Icon(
 | |
|                     Icons.crop_rotate_rounded,
 | |
|                     color: context.themeData.iconTheme.color,
 | |
|                     size: 25,
 | |
|                   ),
 | |
|                   onPressed: () {
 | |
|                     context.pushRoute(
 | |
|                       CropImageRoute(asset: asset, image: image),
 | |
|                     );
 | |
|                   },
 | |
|                 ),
 | |
|                 Text("crop".tr(), style: context.textTheme.displayMedium),
 | |
|               ],
 | |
|             ),
 | |
|             Column(
 | |
|               mainAxisAlignment: MainAxisAlignment.center,
 | |
|               children: <Widget>[
 | |
|                 IconButton(
 | |
|                   icon: Icon(
 | |
|                     Icons.filter,
 | |
|                     color: context.themeData.iconTheme.color,
 | |
|                     size: 25,
 | |
|                   ),
 | |
|                   onPressed: () {
 | |
|                     context.pushRoute(
 | |
|                       FilterImageRoute(
 | |
|                         asset: asset,
 | |
|                         image: image,
 | |
|                       ),
 | |
|                     );
 | |
|                   },
 | |
|                 ),
 | |
|                 Text("filter".tr(), style: context.textTheme.displayMedium),
 | |
|               ],
 | |
|             ),
 | |
|           ],
 | |
|         ),
 | |
|       ),
 | |
|     );
 | |
|   }
 | |
| }
 |