2022-04-02 19:31:53 +02:00
|
|
|
import 'dart:io';
|
|
|
|
|
2022-07-13 14:23:48 +02:00
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2024-02-24 04:38:57 +01:00
|
|
|
import 'package:immich_mobile/extensions/response_extensions.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
2024-09-18 17:15:52 +02:00
|
|
|
import 'package:immich_mobile/interfaces/file_media.interface.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/providers/api.provider.dart';
|
2024-09-18 17:15:52 +02:00
|
|
|
import 'package:immich_mobile/repositories/file_media.repository.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/services/api.service.dart';
|
2023-10-19 22:19:00 +02:00
|
|
|
import 'package:logging/logging.dart';
|
2022-04-02 19:31:53 +02:00
|
|
|
|
|
|
|
import 'package:path_provider/path_provider.dart';
|
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
final imageViewerServiceProvider = Provider(
|
|
|
|
(ref) => ImageViewerService(
|
|
|
|
ref.watch(apiServiceProvider),
|
|
|
|
ref.watch(fileMediaRepositoryProvider),
|
|
|
|
),
|
|
|
|
);
|
2022-07-13 14:23:48 +02:00
|
|
|
|
2022-04-02 19:31:53 +02:00
|
|
|
class ImageViewerService {
|
2022-07-13 14:23:48 +02:00
|
|
|
final ApiService _apiService;
|
2024-09-18 17:15:52 +02:00
|
|
|
final IFileMediaRepository _fileMediaRepository;
|
2023-10-19 22:19:00 +02:00
|
|
|
final Logger _log = Logger("ImageViewerService");
|
2022-08-03 07:04:34 +02:00
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
ImageViewerService(this._apiService, this._fileMediaRepository);
|
2022-07-13 14:23:48 +02:00
|
|
|
|
2024-09-05 19:33:55 +02:00
|
|
|
Future<bool> downloadAsset(Asset asset) async {
|
2023-12-10 16:56:39 +01:00
|
|
|
File? imageFile;
|
|
|
|
File? videoFile;
|
2022-04-02 19:31:53 +02:00
|
|
|
try {
|
2022-11-19 06:12:54 +01:00
|
|
|
// Download LivePhotos image and motion part
|
2023-12-10 16:56:39 +01:00
|
|
|
if (asset.isImage && asset.livePhotoVideoId != null && Platform.isIOS) {
|
2024-01-27 17:14:32 +01:00
|
|
|
var imageResponse =
|
2024-05-31 19:44:04 +02:00
|
|
|
await _apiService.assetsApi.downloadAssetWithHttpInfo(
|
2023-02-04 21:42:42 +01:00
|
|
|
asset.remoteId!,
|
2022-11-19 06:12:54 +01:00
|
|
|
);
|
|
|
|
|
2024-05-31 19:44:04 +02:00
|
|
|
var motionResponse =
|
|
|
|
await _apiService.assetsApi.downloadAssetWithHttpInfo(
|
2022-11-19 06:12:54 +01:00
|
|
|
asset.livePhotoVideoId!,
|
|
|
|
);
|
2022-04-02 19:31:53 +02:00
|
|
|
|
2023-10-19 22:19:00 +02:00
|
|
|
if (imageResponse.statusCode != 200 ||
|
2024-05-31 19:44:04 +02:00
|
|
|
motionResponse.statusCode != 200) {
|
2023-10-19 22:19:00 +02:00
|
|
|
final failedResponse =
|
2024-05-31 19:44:04 +02:00
|
|
|
imageResponse.statusCode != 200 ? imageResponse : motionResponse;
|
2023-10-19 22:19:00 +02:00
|
|
|
_log.severe(
|
2024-02-24 04:38:57 +01:00
|
|
|
"Motion asset download failed",
|
|
|
|
failedResponse.toLoggerString(),
|
2023-10-19 22:19:00 +02:00
|
|
|
);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
Asset? resultAsset;
|
2022-04-02 19:31:53 +02:00
|
|
|
|
2022-11-19 06:12:54 +01:00
|
|
|
final tempDir = await getTemporaryDirectory();
|
2023-12-10 16:56:39 +01:00
|
|
|
videoFile = await File('${tempDir.path}/livephoto.mov').create();
|
|
|
|
imageFile = await File('${tempDir.path}/livephoto.heic').create();
|
2024-05-31 19:44:04 +02:00
|
|
|
videoFile.writeAsBytesSync(motionResponse.bodyBytes);
|
2022-11-19 06:12:54 +01:00
|
|
|
imageFile.writeAsBytesSync(imageResponse.bodyBytes);
|
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
resultAsset = await _fileMediaRepository.saveLivePhoto(
|
|
|
|
image: imageFile,
|
|
|
|
video: videoFile,
|
2023-02-04 21:42:42 +01:00
|
|
|
title: asset.fileName,
|
2022-04-02 19:31:53 +02:00
|
|
|
);
|
2022-11-19 06:12:54 +01:00
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
if (resultAsset == null) {
|
2023-12-10 16:56:39 +01:00
|
|
|
_log.warning(
|
|
|
|
"Asset cannot be saved as a live photo. This is most likely a motion photo. Saving only the image file",
|
|
|
|
);
|
2024-09-18 17:15:52 +02:00
|
|
|
resultAsset = await _fileMediaRepository
|
|
|
|
.saveImage(imageResponse.bodyBytes, title: asset.fileName);
|
2023-12-10 16:56:39 +01:00
|
|
|
}
|
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
return resultAsset != null;
|
2022-04-02 19:31:53 +02:00
|
|
|
} else {
|
2024-05-31 19:44:04 +02:00
|
|
|
var res = await _apiService.assetsApi
|
|
|
|
.downloadAssetWithHttpInfo(asset.remoteId!);
|
2022-04-02 19:31:53 +02:00
|
|
|
|
2023-10-19 22:19:00 +02:00
|
|
|
if (res.statusCode != 200) {
|
2024-02-24 04:38:57 +01:00
|
|
|
_log.severe("Asset download failed", res.toLoggerString());
|
2023-10-19 22:19:00 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-09-18 17:15:52 +02:00
|
|
|
final Asset? resultAsset;
|
2024-09-05 19:33:55 +02:00
|
|
|
final relativePath = Platform.isAndroid ? 'DCIM/Immich' : null;
|
2022-11-19 06:12:54 +01:00
|
|
|
|
2023-02-04 21:42:42 +01:00
|
|
|
if (asset.isImage) {
|
2024-09-18 17:15:52 +02:00
|
|
|
resultAsset = await _fileMediaRepository.saveImage(
|
2022-11-19 06:12:54 +01:00
|
|
|
res.bodyBytes,
|
2023-02-04 21:42:42 +01:00
|
|
|
title: asset.fileName,
|
2024-09-05 19:33:55 +02:00
|
|
|
relativePath: relativePath,
|
2022-11-19 06:12:54 +01:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
final tempDir = await getTemporaryDirectory();
|
2023-12-10 16:56:39 +01:00
|
|
|
videoFile = await File('${tempDir.path}/${asset.fileName}').create();
|
|
|
|
videoFile.writeAsBytesSync(res.bodyBytes);
|
2024-09-18 17:15:52 +02:00
|
|
|
resultAsset = await _fileMediaRepository.saveVideo(
|
2024-09-05 19:33:55 +02:00
|
|
|
videoFile,
|
|
|
|
title: asset.fileName,
|
|
|
|
relativePath: relativePath,
|
|
|
|
);
|
2022-11-19 06:12:54 +01:00
|
|
|
}
|
2024-09-18 17:15:52 +02:00
|
|
|
return resultAsset != null;
|
2022-11-19 06:12:54 +01:00
|
|
|
}
|
2023-11-22 04:55:51 +01:00
|
|
|
} catch (error, stack) {
|
2024-02-24 04:38:57 +01:00
|
|
|
_log.severe("Error saving downloaded asset", error, stack);
|
2022-04-02 19:31:53 +02:00
|
|
|
return false;
|
2023-12-10 16:56:39 +01:00
|
|
|
} finally {
|
|
|
|
// Clear temp files
|
|
|
|
imageFile?.delete();
|
|
|
|
videoFile?.delete();
|
2022-04-02 19:31:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|