mirror of
https://github.com/immich-app/immich.git
synced 2025-01-24 20:52:44 +01:00
960b68b02f
* reverts: 5566 * fix: stitch livePhoto only in iOS * fix: PMProgressHandler only on iOS * ios: fallback to saving image if livephoto fails --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
107 lines
3.5 KiB
Dart
107 lines
3.5 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/shared/models/asset.dart';
|
|
import 'package:immich_mobile/shared/providers/api.provider.dart';
|
|
import 'package:immich_mobile/shared/services/api.service.dart';
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:photo_manager/photo_manager.dart';
|
|
import 'package:path_provider/path_provider.dart';
|
|
|
|
final imageViewerServiceProvider =
|
|
Provider((ref) => ImageViewerService(ref.watch(apiServiceProvider)));
|
|
|
|
class ImageViewerService {
|
|
final ApiService _apiService;
|
|
final Logger _log = Logger("ImageViewerService");
|
|
|
|
ImageViewerService(this._apiService);
|
|
|
|
Future<bool> downloadAssetToDevice(Asset asset) async {
|
|
File? imageFile;
|
|
File? videoFile;
|
|
try {
|
|
// Download LivePhotos image and motion part
|
|
if (asset.isImage && asset.livePhotoVideoId != null && Platform.isIOS) {
|
|
var imageResponse = await _apiService.assetApi.downloadFileWithHttpInfo(
|
|
asset.remoteId!,
|
|
);
|
|
|
|
var motionReponse = await _apiService.assetApi.downloadFileWithHttpInfo(
|
|
asset.livePhotoVideoId!,
|
|
);
|
|
|
|
if (imageResponse.statusCode != 200 ||
|
|
motionReponse.statusCode != 200) {
|
|
final failedResponse =
|
|
imageResponse.statusCode != 200 ? imageResponse : motionReponse;
|
|
_log.severe(
|
|
"Motion asset download failed with status - ${failedResponse.statusCode} and response - ${failedResponse.body}",
|
|
);
|
|
return false;
|
|
}
|
|
|
|
AssetEntity? entity;
|
|
|
|
final tempDir = await getTemporaryDirectory();
|
|
videoFile = await File('${tempDir.path}/livephoto.mov').create();
|
|
imageFile = await File('${tempDir.path}/livephoto.heic').create();
|
|
videoFile.writeAsBytesSync(motionReponse.bodyBytes);
|
|
imageFile.writeAsBytesSync(imageResponse.bodyBytes);
|
|
|
|
entity = await PhotoManager.editor.darwin.saveLivePhoto(
|
|
imageFile: imageFile,
|
|
videoFile: videoFile,
|
|
title: asset.fileName,
|
|
);
|
|
|
|
if (entity == null) {
|
|
_log.warning(
|
|
"Asset cannot be saved as a live photo. This is most likely a motion photo. Saving only the image file",
|
|
);
|
|
|
|
entity = await PhotoManager.editor.saveImage(
|
|
imageResponse.bodyBytes,
|
|
title: asset.fileName,
|
|
);
|
|
}
|
|
|
|
return entity != null;
|
|
} else {
|
|
var res = await _apiService.assetApi
|
|
.downloadFileWithHttpInfo(asset.remoteId!);
|
|
|
|
if (res.statusCode != 200) {
|
|
_log.severe(
|
|
"Asset download failed with status - ${res.statusCode} and response - ${res.body}",
|
|
);
|
|
return false;
|
|
}
|
|
|
|
final AssetEntity? entity;
|
|
|
|
if (asset.isImage) {
|
|
entity = await PhotoManager.editor.saveImage(
|
|
res.bodyBytes,
|
|
title: asset.fileName,
|
|
);
|
|
} else {
|
|
final tempDir = await getTemporaryDirectory();
|
|
videoFile = await File('${tempDir.path}/${asset.fileName}').create();
|
|
videoFile.writeAsBytesSync(res.bodyBytes);
|
|
entity = await PhotoManager.editor
|
|
.saveVideo(videoFile, title: asset.fileName);
|
|
}
|
|
return entity != null;
|
|
}
|
|
} catch (error, stack) {
|
|
_log.severe("Error saving file ${error.toString()}", error, stack);
|
|
return false;
|
|
} finally {
|
|
// Clear temp files
|
|
imageFile?.delete();
|
|
videoFile?.delete();
|
|
}
|
|
}
|
|
}
|