mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-25 07:49:05 -04:00 
			
		
		
		
	refactor(mobile): share action button in new timeline (#19967)
* share asset button * include source * move to repository * formatting
This commit is contained in:
		
							parent
							
								
									531515daf9
								
							
						
					
					
						commit
						055b930066
					
				| @ -1693,6 +1693,7 @@ | ||||
|   "settings_saved": "Settings saved", | ||||
|   "setup_pin_code": "Setup a PIN code", | ||||
|   "share": "Share", | ||||
|   "share_action_prompt": "Shared {count} assets", | ||||
|   "share_add_photos": "Add photos", | ||||
|   "share_assets_selected": "{count} selected", | ||||
|   "share_dialog_preparing": "Preparing...", | ||||
|  | ||||
| @ -1,12 +1,49 @@ | ||||
| import 'dart:io'; | ||||
| 
 | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:fluttertoast/fluttertoast.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:immich_mobile/constants/enums.dart'; | ||||
| import 'package:immich_mobile/extensions/translate_extensions.dart'; | ||||
| import 'package:immich_mobile/presentation/widgets/action_buttons/base_action_button.widget.dart'; | ||||
| import 'package:immich_mobile/providers/infrastructure/action.provider.dart'; | ||||
| import 'package:immich_mobile/providers/timeline/multiselect.provider.dart'; | ||||
| import 'package:immich_mobile/widgets/common/immich_toast.dart'; | ||||
| 
 | ||||
| class ShareActionButton extends ConsumerWidget { | ||||
|   const ShareActionButton({super.key}); | ||||
|   final ActionSource source; | ||||
| 
 | ||||
|   const ShareActionButton({super.key, required this.source}); | ||||
| 
 | ||||
|   void _onTap(BuildContext context, WidgetRef ref) async { | ||||
|     if (!context.mounted) { | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|     final result = await ref.read(actionProvider.notifier).shareAssets(source); | ||||
|     ref.read(multiSelectProvider.notifier).reset(); | ||||
| 
 | ||||
|     if (!context.mounted) { | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|     if (!result.success) { | ||||
|       ImmichToast.show( | ||||
|         context: context, | ||||
|         msg: 'scaffold_body_error_occurred'.t(context: context), | ||||
|         gravity: ToastGravity.BOTTOM, | ||||
|         toastType: ToastType.error, | ||||
|       ); | ||||
|     } else if (result.count > 0) { | ||||
|       ImmichToast.show( | ||||
|         context: context, | ||||
|         msg: 'share_action_prompt' | ||||
|             .t(context: context, args: {'count': result.count.toString()}), | ||||
|         gravity: ToastGravity.BOTTOM, | ||||
|         toastType: ToastType.success, | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context, WidgetRef ref) { | ||||
| @ -14,6 +51,7 @@ class ShareActionButton extends ConsumerWidget { | ||||
|       iconData: | ||||
|           Platform.isAndroid ? Icons.share_rounded : Icons.ios_share_rounded, | ||||
|       label: 'share'.t(context: context), | ||||
|       onPressed: () => _onTap(context, ref), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -38,7 +38,7 @@ class ViewerBottomBar extends ConsumerWidget { | ||||
|     } | ||||
| 
 | ||||
|     final actions = <Widget>[ | ||||
|       const ShareActionButton(), | ||||
|       const ShareActionButton(source: ActionSource.viewer), | ||||
|       const _EditActionButton(), | ||||
|       if (asset.hasRemote && isOwner) | ||||
|         const ArchiveActionButton(source: ActionSource.viewer), | ||||
|  | ||||
| @ -45,7 +45,7 @@ class AssetDetailBottomSheet extends ConsumerWidget { | ||||
|     ); | ||||
| 
 | ||||
|     final actions = <Widget>[ | ||||
|       const ShareActionButton(), | ||||
|       const ShareActionButton(source: ActionSource.viewer), | ||||
|       if (asset.hasRemote) ...[ | ||||
|         const ShareLinkActionButton(source: ActionSource.viewer), | ||||
|         const ArchiveActionButton(source: ActionSource.viewer), | ||||
|  | ||||
| @ -33,7 +33,7 @@ class ArchiveBottomSheet extends ConsumerWidget { | ||||
|       maxChildSize: 0.4, | ||||
|       shouldCloseOnMinExtent: false, | ||||
|       actions: [ | ||||
|         const ShareActionButton(), | ||||
|         const ShareActionButton(source: ActionSource.timeline), | ||||
|         if (multiselect.hasRemote) ...[ | ||||
|           const ShareLinkActionButton(source: ActionSource.timeline), | ||||
|           const UnArchiveActionButton(source: ActionSource.timeline), | ||||
|  | ||||
| @ -33,7 +33,7 @@ class FavoriteBottomSheet extends ConsumerWidget { | ||||
|       maxChildSize: 0.4, | ||||
|       shouldCloseOnMinExtent: false, | ||||
|       actions: [ | ||||
|         const ShareActionButton(), | ||||
|         const ShareActionButton(source: ActionSource.timeline), | ||||
|         if (multiselect.hasRemote) ...[ | ||||
|           const ShareLinkActionButton(source: ActionSource.timeline), | ||||
|           const UnFavoriteActionButton(source: ActionSource.timeline), | ||||
|  | ||||
| @ -33,7 +33,7 @@ class GeneralBottomSheet extends ConsumerWidget { | ||||
|       maxChildSize: 0.4, | ||||
|       shouldCloseOnMinExtent: false, | ||||
|       actions: [ | ||||
|         const ShareActionButton(), | ||||
|         const ShareActionButton(source: ActionSource.timeline), | ||||
|         if (multiselect.hasRemote) ...[ | ||||
|           const ShareLinkActionButton(source: ActionSource.timeline), | ||||
|           const ArchiveActionButton(source: ActionSource.timeline), | ||||
|  | ||||
| @ -16,7 +16,7 @@ class LocalAlbumBottomSheet extends ConsumerWidget { | ||||
|       maxChildSize: 0.4, | ||||
|       shouldCloseOnMinExtent: false, | ||||
|       actions: [ | ||||
|         ShareActionButton(), | ||||
|         ShareActionButton(source: ActionSource.timeline), | ||||
|         DeleteLocalActionButton(source: ActionSource.timeline), | ||||
|         UploadActionButton(), | ||||
|       ], | ||||
|  | ||||
| @ -17,7 +17,7 @@ class LockedFolderBottomSheet extends ConsumerWidget { | ||||
|       maxChildSize: 0.4, | ||||
|       shouldCloseOnMinExtent: false, | ||||
|       actions: [ | ||||
|         ShareActionButton(), | ||||
|         ShareActionButton(source: ActionSource.timeline), | ||||
|         DownloadActionButton(), | ||||
|         DeletePermanentActionButton(source: ActionSource.timeline), | ||||
|         RemoveFromLockFolderActionButton(source: ActionSource.timeline), | ||||
|  | ||||
| @ -1,5 +1,6 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:immich_mobile/constants/enums.dart'; | ||||
| import 'package:immich_mobile/presentation/widgets/action_buttons/download_action_button.widget.dart'; | ||||
| import 'package:immich_mobile/presentation/widgets/action_buttons/share_action_button.widget.dart'; | ||||
| import 'package:immich_mobile/presentation/widgets/bottom_sheet/base_bottom_sheet.widget.dart'; | ||||
| @ -14,7 +15,7 @@ class PartnerDetailBottomSheet extends ConsumerWidget { | ||||
|       maxChildSize: 0.4, | ||||
|       shouldCloseOnMinExtent: false, | ||||
|       actions: [ | ||||
|         ShareActionButton(), | ||||
|         ShareActionButton(source: ActionSource.timeline), | ||||
|         DownloadActionButton(), | ||||
|       ], | ||||
|     ); | ||||
|  | ||||
| @ -36,7 +36,7 @@ class RemoteAlbumBottomSheet extends ConsumerWidget { | ||||
|       maxChildSize: 0.4, | ||||
|       shouldCloseOnMinExtent: false, | ||||
|       actions: [ | ||||
|         const ShareActionButton(), | ||||
|         const ShareActionButton(source: ActionSource.timeline), | ||||
|         if (multiselect.hasRemote) ...[ | ||||
|           const ShareLinkActionButton(source: ActionSource.timeline), | ||||
|           const ArchiveActionButton(source: ActionSource.timeline), | ||||
|  | ||||
| @ -58,15 +58,18 @@ class ActionNotifier extends Notifier<void> { | ||||
|         .toList(growable: false); | ||||
|   } | ||||
| 
 | ||||
|   Iterable<T> _getIdsForSource<T extends BaseAsset>(ActionSource source) { | ||||
|     final Set<BaseAsset> assets = switch (source) { | ||||
|   Set<BaseAsset> _getAssets(ActionSource source) { | ||||
|     return switch (source) { | ||||
|       ActionSource.timeline => ref.read(multiSelectProvider).selectedAssets, | ||||
|       ActionSource.viewer => switch (ref.read(currentAssetNotifier)) { | ||||
|           BaseAsset asset => {asset}, | ||||
|           null => const {}, | ||||
|         }, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   Iterable<T> _getIdsForSource<T extends BaseAsset>(ActionSource source) { | ||||
|     final Set<BaseAsset> assets = _getAssets(source); | ||||
|     return switch (T) { | ||||
|       const (RemoteAsset) => assets.whereType<RemoteAsset>(), | ||||
|       const (LocalAsset) => assets.whereType<LocalAsset>(), | ||||
| @ -266,6 +269,22 @@ class ActionNotifier extends Notifier<void> { | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Future<ActionResult> shareAssets(ActionSource source) async { | ||||
|     final ids = _getAssets(source).toList(growable: false); | ||||
| 
 | ||||
|     try { | ||||
|       final count = await _service.shareAssets(ids); | ||||
|       return ActionResult(count: count, success: true); | ||||
|     } catch (error, stack) { | ||||
|       _logger.severe('Failed to share assets', error, stack); | ||||
|       return ActionResult( | ||||
|         count: ids.length, | ||||
|         success: false, | ||||
|         error: error.toString(), | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| extension on Iterable<RemoteAsset> { | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:http/http.dart'; | ||||
| import 'package:immich_mobile/constants/enums.dart'; | ||||
| import 'package:immich_mobile/entities/asset.entity.dart'; | ||||
| import 'package:immich_mobile/providers/api.provider.dart'; | ||||
| @ -83,6 +84,10 @@ class AssetApiRepository extends ApiRepository { | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   Future<Response> downloadAsset(String id) { | ||||
|     return _api.downloadAssetWithHttpInfo(id); | ||||
|   } | ||||
| 
 | ||||
|   _mapVisibility(AssetVisibilityEnum visibility) => switch (visibility) { | ||||
|         AssetVisibilityEnum.timeline => AssetVisibility.timeline, | ||||
|         AssetVisibilityEnum.hidden => AssetVisibility.hidden, | ||||
|  | ||||
| @ -1,27 +1,40 @@ | ||||
| import 'dart:io'; | ||||
| 
 | ||||
| import 'package:hooks_riverpod/hooks_riverpod.dart'; | ||||
| import 'package:immich_mobile/domain/models/exif.model.dart'; | ||||
| import 'package:immich_mobile/domain/models/store.model.dart'; | ||||
| import 'package:immich_mobile/entities/asset.entity.dart'; | ||||
| import 'package:immich_mobile/entities/asset.entity.dart' as asset_entity; | ||||
| import 'package:immich_mobile/entities/store.entity.dart'; | ||||
| import 'package:immich_mobile/repositories/asset_api.repository.dart'; | ||||
| import 'package:immich_mobile/utils/hash.dart'; | ||||
| import 'package:photo_manager/photo_manager.dart' hide AssetType; | ||||
| import 'package:logging/logging.dart'; | ||||
| import 'package:path_provider/path_provider.dart'; | ||||
| import 'package:photo_manager/photo_manager.dart'; | ||||
| import 'package:immich_mobile/domain/models/asset/base_asset.model.dart'; | ||||
| import 'package:immich_mobile/extensions/response_extensions.dart'; | ||||
| import 'package:share_plus/share_plus.dart'; | ||||
| 
 | ||||
| final assetMediaRepositoryProvider = | ||||
|     Provider((ref) => const AssetMediaRepository()); | ||||
| final assetMediaRepositoryProvider = Provider( | ||||
|   (ref) => AssetMediaRepository(ref.watch(assetApiRepositoryProvider)), | ||||
| ); | ||||
| 
 | ||||
| class AssetMediaRepository { | ||||
|   const AssetMediaRepository(); | ||||
|   final AssetApiRepository _assetApiRepository; | ||||
|   static final Logger _log = Logger("AssetMediaRepository"); | ||||
| 
 | ||||
|   const AssetMediaRepository(this._assetApiRepository); | ||||
| 
 | ||||
|   Future<List<String>> deleteAll(List<String> ids) => | ||||
|       PhotoManager.editor.deleteWithIds(ids); | ||||
| 
 | ||||
|   Future<Asset?> get(String id) async { | ||||
|   Future<asset_entity.Asset?> get(String id) async { | ||||
|     final entity = await AssetEntity.fromId(id); | ||||
|     return toAsset(entity); | ||||
|   } | ||||
| 
 | ||||
|   static Asset? toAsset(AssetEntity? local) { | ||||
|   static asset_entity.Asset? toAsset(AssetEntity? local) { | ||||
|     if (local == null) return null; | ||||
|     final Asset asset = Asset( | ||||
|     final asset_entity.Asset asset = asset_entity.Asset( | ||||
|       checksum: "", | ||||
|       localId: local.id, | ||||
|       ownerId: fastHash(Store.get(StoreKey.currentUser).id), | ||||
| @ -29,7 +42,7 @@ class AssetMediaRepository { | ||||
|       fileModifiedAt: local.modifiedDateTime, | ||||
|       updatedAt: local.modifiedDateTime, | ||||
|       durationInSeconds: local.duration, | ||||
|       type: AssetType.values[local.typeInt], | ||||
|       type: asset_entity.AssetType.values[local.typeInt], | ||||
|       fileName: local.title!, | ||||
|       width: local.width, | ||||
|       height: local.height, | ||||
| @ -57,4 +70,57 @@ class AssetMediaRepository { | ||||
|     // otherwise using the `entity.title` would return a random GUID | ||||
|     return await entity.titleAsync; | ||||
|   } | ||||
| 
 | ||||
|   // TODO: make this more efficient | ||||
|   Future<int> shareAssets(List<BaseAsset> assets) async { | ||||
|     final downloadedXFiles = <XFile>[]; | ||||
| 
 | ||||
|     for (var asset in assets) { | ||||
|       final localId = (asset is LocalAsset) | ||||
|           ? asset.id | ||||
|           : asset is RemoteAsset | ||||
|               ? asset.localId | ||||
|               : null; | ||||
|       if (localId != null) { | ||||
|         File? f = | ||||
|             await AssetEntity(id: localId, width: 1, height: 1, typeInt: 0) | ||||
|                 .originFile; | ||||
|         downloadedXFiles.add(XFile(f!.path)); | ||||
|       } else if (asset is RemoteAsset) { | ||||
|         final tempDir = await getTemporaryDirectory(); | ||||
|         final name = asset.name; | ||||
|         final tempFile = await File('${tempDir.path}/$name').create(); | ||||
|         final res = await _assetApiRepository.downloadAsset(asset.id); | ||||
| 
 | ||||
|         if (res.statusCode != 200) { | ||||
|           _log.severe("Download for $name failed", res.toLoggerString()); | ||||
|           continue; | ||||
|         } | ||||
| 
 | ||||
|         await tempFile.writeAsBytes(res.bodyBytes); | ||||
|         downloadedXFiles.add(XFile(tempFile.path)); | ||||
|       } else { | ||||
|         _log.warning("Asset type not supported for sharing: $asset"); | ||||
|         continue; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (downloadedXFiles.isEmpty) { | ||||
|       _log.warning("No asset can be retrieved for share"); | ||||
|       return 0; | ||||
|     } | ||||
| 
 | ||||
|     final result = await Share.shareXFiles(downloadedXFiles); | ||||
| 
 | ||||
|     for (var file in downloadedXFiles) { | ||||
|       try { | ||||
|         await File(file.path).delete(); | ||||
|       } catch (e) { | ||||
|         _log.warning("Failed to delete temporary file: ${file.path}", e); | ||||
|       } | ||||
|     } | ||||
|     return result.status == ShareResultStatus.success | ||||
|         ? downloadedXFiles.length | ||||
|         : 0; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -12,7 +12,7 @@ import 'package:immich_mobile/repositories/asset_media.repository.dart'; | ||||
| import 'package:immich_mobile/repositories/drift_album_api_repository.dart'; | ||||
| import 'package:immich_mobile/routing/router.dart'; | ||||
| import 'package:immich_mobile/widgets/common/location_picker.dart'; | ||||
| import 'package:maplibre_gl/maplibre_gl.dart'; | ||||
| import 'package:maplibre_gl/maplibre_gl.dart' as maplibre; | ||||
| import 'package:riverpod_annotation/riverpod_annotation.dart'; | ||||
| 
 | ||||
| final actionServiceProvider = Provider<ActionService>( | ||||
| @ -124,12 +124,12 @@ class ActionService { | ||||
|     List<String> remoteIds, | ||||
|     BuildContext context, | ||||
|   ) async { | ||||
|     LatLng? initialLatLng; | ||||
|     maplibre.LatLng? initialLatLng; | ||||
|     if (remoteIds.length == 1) { | ||||
|       final exif = await _remoteAssetRepository.getExif(remoteIds[0]); | ||||
| 
 | ||||
|       if (exif?.latitude != null && exif?.longitude != null) { | ||||
|         initialLatLng = LatLng(exif!.latitude!, exif.longitude!); | ||||
|         initialLatLng = maplibre.LatLng(exif!.latitude!, exif.longitude!); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
| @ -165,4 +165,8 @@ class ActionService { | ||||
| 
 | ||||
|     return removedCount; | ||||
|   } | ||||
| 
 | ||||
|   Future<int> shareAssets(List<BaseAsset> assets) { | ||||
|     return _assetMediaRepository.shareAssets(assets); | ||||
|   } | ||||
| } | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user