2024-02-13 21:30:32 +00:00
|
|
|
import 'dart:async';
|
|
|
|
import 'dart:ui' as ui;
|
|
|
|
|
|
|
|
import 'package:cached_network_image/cached_network_image.dart';
|
2024-03-14 20:29:09 +00:00
|
|
|
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
2024-05-02 20:59:14 +00:00
|
|
|
import 'package:immich_mobile/providers/image/cache/image_loader.dart';
|
|
|
|
import 'package:immich_mobile/providers/image/cache/remote_image_cache_manager.dart';
|
2024-02-13 21:30:32 +00:00
|
|
|
import 'package:openapi/api.dart' as api;
|
|
|
|
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:flutter/painting.dart';
|
2024-05-02 20:59:14 +00:00
|
|
|
import 'package:immich_mobile/services/app_settings.service.dart';
|
2024-05-01 02:36:40 +00:00
|
|
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
|
|
|
import 'package:immich_mobile/entities/store.entity.dart';
|
2024-02-13 21:30:32 +00:00
|
|
|
import 'package:immich_mobile/utils/image_url_builder.dart';
|
|
|
|
|
2024-03-14 20:29:09 +00:00
|
|
|
/// The remote image provider for full size remote images
|
2024-02-27 15:51:19 +00:00
|
|
|
class ImmichRemoteImageProvider
|
|
|
|
extends ImageProvider<ImmichRemoteImageProvider> {
|
2024-02-13 21:30:32 +00:00
|
|
|
/// The [Asset.remoteId] of the asset to fetch
|
|
|
|
final String assetId;
|
|
|
|
|
2024-03-14 20:29:09 +00:00
|
|
|
/// The image cache manager
|
2024-04-25 04:30:32 +00:00
|
|
|
final CacheManager? cacheManager;
|
2024-02-13 21:30:32 +00:00
|
|
|
|
|
|
|
ImmichRemoteImageProvider({
|
|
|
|
required this.assetId,
|
2024-03-14 20:29:09 +00:00
|
|
|
this.cacheManager,
|
2024-02-13 21:30:32 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
/// Converts an [ImageProvider]'s settings plus an [ImageConfiguration] to a key
|
|
|
|
/// that describes the precise image to load.
|
|
|
|
@override
|
2024-02-27 15:51:19 +00:00
|
|
|
Future<ImmichRemoteImageProvider> obtainKey(
|
|
|
|
ImageConfiguration configuration,
|
|
|
|
) {
|
|
|
|
return SynchronousFuture(this);
|
2024-02-13 21:30:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
2024-02-27 15:51:19 +00:00
|
|
|
ImageStreamCompleter loadImage(
|
|
|
|
ImmichRemoteImageProvider key,
|
|
|
|
ImageDecoderCallback decode,
|
|
|
|
) {
|
2024-03-14 20:29:09 +00:00
|
|
|
final cache = cacheManager ?? RemoteImageCacheManager();
|
2024-02-13 21:30:32 +00:00
|
|
|
final chunkEvents = StreamController<ImageChunkEvent>();
|
|
|
|
return MultiImageStreamCompleter(
|
2024-03-14 20:29:09 +00:00
|
|
|
codec: _codec(key, cache, decode, chunkEvents),
|
2024-02-13 21:30:32 +00:00
|
|
|
scale: 1.0,
|
|
|
|
chunkEvents: chunkEvents.stream,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Whether to show the original file or load a compressed version
|
|
|
|
bool get _useOriginal => Store.get(
|
|
|
|
AppSettingsEnum.loadOriginal.storeKey,
|
|
|
|
AppSettingsEnum.loadOriginal.defaultValue,
|
|
|
|
);
|
|
|
|
|
|
|
|
/// Whether to load the preview thumbnail first or not
|
|
|
|
bool get _loadPreview => Store.get(
|
|
|
|
AppSettingsEnum.loadPreview.storeKey,
|
|
|
|
AppSettingsEnum.loadPreview.defaultValue,
|
|
|
|
);
|
|
|
|
|
|
|
|
// Streams in each stage of the image as we ask for it
|
|
|
|
Stream<ui.Codec> _codec(
|
2024-02-27 15:51:19 +00:00
|
|
|
ImmichRemoteImageProvider key,
|
2024-04-25 04:30:32 +00:00
|
|
|
CacheManager cache,
|
2024-02-13 21:30:32 +00:00
|
|
|
ImageDecoderCallback decode,
|
|
|
|
StreamController<ImageChunkEvent> chunkEvents,
|
|
|
|
) async* {
|
|
|
|
// Load a preview to the chunk events
|
2024-03-14 20:29:09 +00:00
|
|
|
if (_loadPreview) {
|
2024-02-13 21:30:32 +00:00
|
|
|
final preview = getThumbnailUrlForRemoteId(
|
2024-02-27 15:51:19 +00:00
|
|
|
key.assetId,
|
2024-05-31 17:44:04 +00:00
|
|
|
type: api.AssetMediaSize.thumbnail,
|
2024-02-13 21:30:32 +00:00
|
|
|
);
|
|
|
|
|
2024-03-14 20:29:09 +00:00
|
|
|
yield await ImageLoader.loadImageFromCache(
|
|
|
|
preview,
|
|
|
|
cache: cache,
|
|
|
|
decode: decode,
|
|
|
|
chunkEvents: chunkEvents,
|
2024-02-13 21:30:32 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load the higher resolution version of the image
|
|
|
|
final url = getThumbnailUrlForRemoteId(
|
2024-02-27 15:51:19 +00:00
|
|
|
key.assetId,
|
2024-05-31 17:44:04 +00:00
|
|
|
type: api.AssetMediaSize.preview,
|
2024-02-13 21:30:32 +00:00
|
|
|
);
|
2024-03-14 20:29:09 +00:00
|
|
|
final codec = await ImageLoader.loadImageFromCache(
|
|
|
|
url,
|
|
|
|
cache: cache,
|
|
|
|
decode: decode,
|
|
|
|
chunkEvents: chunkEvents,
|
|
|
|
);
|
2024-02-13 21:30:32 +00:00
|
|
|
yield codec;
|
|
|
|
|
|
|
|
// Load the final remote image
|
|
|
|
if (_useOriginal) {
|
|
|
|
// Load the original image
|
2024-08-14 19:52:19 +00:00
|
|
|
final url = getOriginalUrlForRemoteId(key.assetId);
|
2024-03-14 20:29:09 +00:00
|
|
|
final codec = await ImageLoader.loadImageFromCache(
|
|
|
|
url,
|
|
|
|
cache: cache,
|
|
|
|
decode: decode,
|
|
|
|
chunkEvents: chunkEvents,
|
|
|
|
);
|
2024-02-13 21:30:32 +00:00
|
|
|
yield codec;
|
|
|
|
}
|
|
|
|
await chunkEvents.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool operator ==(Object other) {
|
|
|
|
if (identical(this, other)) return true;
|
2024-03-14 20:29:09 +00:00
|
|
|
if (other is ImmichRemoteImageProvider) {
|
|
|
|
return assetId == other.assetId;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2024-02-13 21:30:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
int get hashCode => assetId.hashCode;
|
|
|
|
}
|