immich/mobile/lib/presentation/widgets/images/local_image_provider.dart

115 lines
3.5 KiB
Dart
Raw Normal View History

import 'dart:async';
import 'dart:ui';
import 'package:flutter/foundation.dart';
import 'package:flutter/widgets.dart';
import 'package:immich_mobile/infrastructure/repositories/asset_media.repository.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/presentation/widgets/timeline/constants.dart';
2025-08-13 00:10:55 -04:00
class LocalThumbProvider extends ImageProvider<LocalThumbProvider> with CancellableImageProviderMixin {
final String id;
final Size size;
2025-08-13 00:10:55 -04:00
LocalThumbProvider({required this.id, this.size = kThumbnailResolution});
@override
Future<LocalThumbProvider> obtainKey(ImageConfiguration configuration) {
return SynchronousFuture(this);
}
@override
ImageStreamCompleter loadImage(LocalThumbProvider key, ImageDecoderCallback decode) {
2025-08-13 00:15:27 -04:00
final completer = OneFramePlaceholderImageStreamCompleter(
2025-08-13 00:10:55 -04:00
_codec(key, decode),
informationCollector: () => <DiagnosticsNode>[
DiagnosticsProperty<String>('Id', key.id),
DiagnosticsProperty<Size>('Size', key.size),
],
);
2025-08-13 00:15:27 -04:00
completer.addOnLastListenerRemovedCallback(cancel);
return completer;
}
2025-08-13 00:10:55 -04:00
Stream<ImageInfo> _codec(LocalThumbProvider key, ImageDecoderCallback decode) async* {
final request = this.request = LocalImageRequest(localId: key.id, size: size);
try {
final image = await request.load(decode);
if (image != null) {
yield image;
}
} finally {
this.request = null;
}
}
@override
bool operator ==(Object other) {
if (identical(this, other)) return true;
if (other is LocalThumbProvider) {
2025-08-13 00:10:55 -04:00
return id == other.id && size == other.size;
}
return false;
}
@override
2025-08-13 00:10:55 -04:00
int get hashCode => id.hashCode ^ size.hashCode;
}
2025-08-13 00:10:55 -04:00
class LocalFullImageProvider extends ImageProvider<LocalFullImageProvider> with CancellableImageProviderMixin {
final String id;
final Size size;
2025-08-13 00:10:55 -04:00
LocalFullImageProvider({required this.id, required this.size});
@override
Future<LocalFullImageProvider> obtainKey(ImageConfiguration configuration) {
return SynchronousFuture(this);
}
@override
ImageStreamCompleter loadImage(LocalFullImageProvider key, ImageDecoderCallback decode) {
2025-08-13 00:15:27 -04:00
final completer = OneFramePlaceholderImageStreamCompleter(
_codec(key, decode),
2025-08-13 00:10:55 -04:00
initialImage: getCachedImage(LocalThumbProvider(id: key.id)),
informationCollector: () => <DiagnosticsNode>[
2025-08-13 00:10:55 -04:00
DiagnosticsProperty<ImageProvider>('Image provider', this),
DiagnosticsProperty<String>('Id', key.id),
DiagnosticsProperty<Size>('Size', key.size),
],
);
2025-08-13 00:15:27 -04:00
completer.addOnLastListenerRemovedCallback(cancel);
return completer;
}
2025-08-13 00:10:55 -04:00
Stream<ImageInfo> _codec(LocalFullImageProvider key, ImageDecoderCallback decode) async* {
final devicePixelRatio = PlatformDispatcher.instance.views.first.devicePixelRatio;
2025-08-13 00:10:55 -04:00
final request = this.request = LocalImageRequest(
localId: key.id,
size: Size(size.width * devicePixelRatio, size.height * devicePixelRatio),
);
2025-08-13 00:10:55 -04:00
try {
final image = await request.load(decode);
if (image != null) {
yield image;
}
2025-08-13 00:10:55 -04:00
} finally {
this.request = null;
}
}
@override
bool operator ==(Object other) {
if (identical(this, other)) return true;
if (other is LocalFullImageProvider) {
2025-08-13 00:10:55 -04:00
return id == other.id && size == other.size;
}
return false;
}
@override
2025-08-13 00:10:55 -04:00
int get hashCode => id.hashCode ^ size.hashCode;
}