mirror of
https://github.com/immich-app/immich.git
synced 2025-10-24 23:39:03 -04:00
* platform image providers * use key * fix cache manager * more logs, cancel on dispose instead * split into separate files * fix saving to cache * cancel multi-stage provider * refactored `getInitialImage` * only wait for disposal for full images * cached image works * formatting * lower asset viewer ram usage --------- Co-authored-by: Alex <alex.tran1502@gmail.com>
133 lines
4.0 KiB
Dart
133 lines
4.0 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/painting.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/providers/image/cache/remote_image_cache_manager.dart';
|
|
import 'package:immich_mobile/services/api.service.dart';
|
|
import 'package:immich_mobile/utils/image_url_builder.dart';
|
|
|
|
class RemoteThumbProvider extends CancellableImageProvider<RemoteThumbProvider>
|
|
with CancellableImageProviderMixin<RemoteThumbProvider> {
|
|
static final cacheManager = RemoteThumbnailCacheManager();
|
|
final String assetId;
|
|
|
|
RemoteThumbProvider({required this.assetId});
|
|
|
|
@override
|
|
Future<RemoteThumbProvider> obtainKey(ImageConfiguration configuration) {
|
|
return SynchronousFuture(this);
|
|
}
|
|
|
|
@override
|
|
ImageStreamCompleter loadImage(RemoteThumbProvider key, ImageDecoderCallback decode) {
|
|
return OneFramePlaceholderImageStreamCompleter(
|
|
_codec(key, decode),
|
|
informationCollector: () => <DiagnosticsNode>[
|
|
DiagnosticsProperty<ImageProvider>('Image provider', this),
|
|
DiagnosticsProperty<String>('Asset Id', key.assetId),
|
|
],
|
|
)..addOnLastListenerRemovedCallback(cancel);
|
|
}
|
|
|
|
Stream<ImageInfo> _codec(RemoteThumbProvider key, ImageDecoderCallback decode) {
|
|
final request = RemoteImageRequest(
|
|
uri: getThumbnailUrlForRemoteId(key.assetId),
|
|
headers: ApiService.getRequestHeaders(),
|
|
cacheManager: cacheManager,
|
|
);
|
|
return loadRequest(request, decode);
|
|
}
|
|
|
|
@override
|
|
bool operator ==(Object other) {
|
|
if (identical(this, other)) return true;
|
|
if (other is RemoteThumbProvider) {
|
|
return assetId == other.assetId;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
@override
|
|
int get hashCode => assetId.hashCode;
|
|
}
|
|
|
|
class RemoteFullImageProvider extends CancellableImageProvider<RemoteFullImageProvider>
|
|
with CancellableImageProviderMixin<RemoteFullImageProvider> {
|
|
static final cacheManager = RemoteThumbnailCacheManager();
|
|
final String assetId;
|
|
|
|
RemoteFullImageProvider({required this.assetId});
|
|
|
|
@override
|
|
Future<RemoteFullImageProvider> obtainKey(ImageConfiguration configuration) {
|
|
return SynchronousFuture(this);
|
|
}
|
|
|
|
@override
|
|
ImageStreamCompleter loadImage(RemoteFullImageProvider key, ImageDecoderCallback decode) {
|
|
return OneFramePlaceholderImageStreamCompleter(
|
|
_codec(key, decode),
|
|
initialImage: getCachedImage(RemoteThumbProvider(assetId: key.assetId)),
|
|
informationCollector: () => <DiagnosticsNode>[
|
|
DiagnosticsProperty<ImageProvider>('Image provider', this),
|
|
DiagnosticsProperty<String>('Asset Id', key.assetId),
|
|
],
|
|
onDispose: cancel,
|
|
);
|
|
}
|
|
|
|
Stream<ImageInfo> _codec(RemoteFullImageProvider key, ImageDecoderCallback decode) async* {
|
|
yield* initialImageStream();
|
|
|
|
if (isCancelled) {
|
|
evict();
|
|
return;
|
|
}
|
|
|
|
final headers = ApiService.getRequestHeaders();
|
|
try {
|
|
final request = RemoteImageRequest(
|
|
uri: getPreviewUrlForRemoteId(key.assetId),
|
|
headers: headers,
|
|
cacheManager: cacheManager,
|
|
);
|
|
yield* loadRequest(request, decode);
|
|
} finally {
|
|
request = null;
|
|
}
|
|
|
|
if (isCancelled) {
|
|
evict();
|
|
return;
|
|
}
|
|
|
|
if (AppSetting.get(Setting.loadOriginal)) {
|
|
try {
|
|
final request = RemoteImageRequest(uri: getOriginalUrlForRemoteId(key.assetId), headers: headers);
|
|
yield* loadRequest(request, decode);
|
|
} finally {
|
|
request = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
@override
|
|
bool operator ==(Object other) {
|
|
if (identical(this, other)) return true;
|
|
if (other is RemoteFullImageProvider) {
|
|
return assetId == other.assetId;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
@override
|
|
int get hashCode => assetId.hashCode;
|
|
}
|