import 'package:flutter/foundation.dart'; import 'package:flutter/painting.dart'; import 'package:immich_mobile/domain/models/asset/base_asset.model.dart'; import 'package:immich_mobile/domain/models/setting.model.dart'; import 'package:immich_mobile/domain/services/setting.service.dart'; import 'package:immich_mobile/infrastructure/loaders/image_request.dart'; import 'package:immich_mobile/presentation/widgets/images/image_provider.dart'; import 'package:immich_mobile/presentation/widgets/images/one_frame_multi_image_stream_completer.dart'; import 'package:immich_mobile/services/api.service.dart'; import 'package:immich_mobile/utils/image_url_builder.dart'; import 'package:openapi/api.dart'; class RemoteImageProvider extends CancellableImageProvider with CancellableImageProviderMixin { final String url; RemoteImageProvider({required this.url}); RemoteImageProvider.thumbnail({required String assetId, required String thumbhash}) : url = getThumbnailUrlForRemoteId(assetId, thumbhash: thumbhash); @override Future obtainKey(ImageConfiguration configuration) { return SynchronousFuture(this); } @override ImageStreamCompleter loadImage(RemoteImageProvider key, ImageDecoderCallback decode) { return OneFramePlaceholderImageStreamCompleter( _codec(key, decode), informationCollector: () => [ DiagnosticsProperty('Image provider', this), DiagnosticsProperty('URL', key.url), ], onLastListenerRemoved: cancel, ); } Stream _codec(RemoteImageProvider key, ImageDecoderCallback decode) { final request = this.request = RemoteImageRequest(uri: key.url, headers: ApiService.getRequestHeaders()); return loadRequest(request, decode); } @override bool operator ==(Object other) { if (identical(this, other)) return true; if (other is RemoteImageProvider) { return url == other.url; } return false; } @override int get hashCode => url.hashCode; } class RemoteFullImageProvider extends CancellableImageProvider with CancellableImageProviderMixin { final String assetId; final String thumbhash; final AssetType assetType; RemoteFullImageProvider({required this.assetId, required this.thumbhash, required this.assetType}); @override Future obtainKey(ImageConfiguration configuration) { return SynchronousFuture(this); } @override ImageStreamCompleter loadImage(RemoteFullImageProvider key, ImageDecoderCallback decode) { return OneFramePlaceholderImageStreamCompleter( _codec(key, decode), initialImage: getInitialImage(RemoteImageProvider.thumbnail(assetId: key.assetId, thumbhash: key.thumbhash)), informationCollector: () => [ DiagnosticsProperty('Image provider', this), DiagnosticsProperty('Asset Id', key.assetId), ], onLastListenerRemoved: cancel, ); } Stream _codec(RemoteFullImageProvider key, ImageDecoderCallback decode) async* { yield* initialImageStream(); if (isCancelled) { PaintingBinding.instance.imageCache.evict(this); return; } final headers = ApiService.getRequestHeaders(); final previewRequest = request = RemoteImageRequest( uri: getThumbnailUrlForRemoteId(key.assetId, type: AssetMediaSize.preview, thumbhash: key.thumbhash), headers: headers, ); yield* loadRequest(previewRequest, decode); if (assetType != AssetType.image || !AppSetting.get(Setting.loadOriginal)) { return; } if (isCancelled) { PaintingBinding.instance.imageCache.evict(this); return; } final originalRequest = request = RemoteImageRequest(uri: getOriginalUrlForRemoteId(key.assetId), headers: headers); yield* loadRequest(originalRequest, decode); } @override bool operator ==(Object other) { if (identical(this, other)) return true; if (other is RemoteFullImageProvider) { return assetId == other.assetId && thumbhash == other.thumbhash; } return false; } @override int get hashCode => assetId.hashCode ^ thumbhash.hashCode; }