mirror of
https://github.com/immich-app/immich.git
synced 2026-04-17 08:01:55 -04:00
* feat: mobile editing fix: openapi patch this sucks :pepehands: chore: migrate some changes from the filtering PR chore: color tweak fix: hide edit button on server versions chore: translation * chore: code review changes chore: code review * sealed class * const constant * enum * concurrent queries * chore: major cleanup to use riverpod provider * fix: aspect ratio selection * chore: typesafe websocket event parsing * fix: wrong disable state for save button * chore: remove isCancelled shim * chore: cleanup postframe callback usage * chore: clean import --------- Co-authored-by: mertalev <101130780+mertalev@users.noreply.github.com>
192 lines
6.2 KiB
Dart
192 lines
6.2 KiB
Dart
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/animated_image_stream_completer.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/utils/image_url_builder.dart';
|
|
import 'package:openapi/api.dart';
|
|
|
|
class RemoteImageProvider extends CancellableImageProvider<RemoteImageProvider>
|
|
with CancellableImageProviderMixin<RemoteImageProvider> {
|
|
final String url;
|
|
final bool edited;
|
|
|
|
RemoteImageProvider({required this.url, this.edited = true});
|
|
|
|
RemoteImageProvider.thumbnail({required String assetId, required String thumbhash, this.edited = true})
|
|
: url = getThumbnailUrlForRemoteId(assetId, thumbhash: thumbhash, edited: edited);
|
|
|
|
@override
|
|
Future<RemoteImageProvider> obtainKey(ImageConfiguration configuration) {
|
|
return SynchronousFuture(this);
|
|
}
|
|
|
|
@override
|
|
ImageStreamCompleter loadImage(RemoteImageProvider key, ImageDecoderCallback decode) {
|
|
return OneFramePlaceholderImageStreamCompleter(
|
|
_codec(key, decode),
|
|
informationCollector: () => <DiagnosticsNode>[
|
|
DiagnosticsProperty<ImageProvider>('Image provider', this),
|
|
DiagnosticsProperty<String>('URL', key.url),
|
|
],
|
|
onLastListenerRemoved: cancel,
|
|
);
|
|
}
|
|
|
|
Stream<ImageInfo> _codec(RemoteImageProvider key, ImageDecoderCallback decode) {
|
|
final request = this.request = RemoteImageRequest(uri: key.url);
|
|
return loadRequest(request, decode, isFinal: true);
|
|
}
|
|
|
|
@override
|
|
bool operator ==(Object other) {
|
|
if (identical(this, other)) return true;
|
|
if (other is RemoteImageProvider) {
|
|
return url == other.url && edited == other.edited;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@override
|
|
int get hashCode => url.hashCode ^ edited.hashCode;
|
|
}
|
|
|
|
class RemoteFullImageProvider extends CancellableImageProvider<RemoteFullImageProvider>
|
|
with CancellableImageProviderMixin<RemoteFullImageProvider> {
|
|
final String assetId;
|
|
final String thumbhash;
|
|
final AssetType assetType;
|
|
final bool isAnimated;
|
|
final bool edited;
|
|
|
|
RemoteFullImageProvider({
|
|
required this.assetId,
|
|
required this.thumbhash,
|
|
required this.assetType,
|
|
required this.isAnimated,
|
|
this.edited = true,
|
|
});
|
|
|
|
@override
|
|
Future<RemoteFullImageProvider> obtainKey(ImageConfiguration configuration) {
|
|
return SynchronousFuture(this);
|
|
}
|
|
|
|
@override
|
|
ImageStreamCompleter loadImage(RemoteFullImageProvider key, ImageDecoderCallback decode) {
|
|
if (key.isAnimated) {
|
|
return AnimatedImageStreamCompleter(
|
|
stream: _animatedCodec(key, decode),
|
|
scale: 1.0,
|
|
initialImage: getInitialImage(RemoteImageProvider.thumbnail(assetId: key.assetId, thumbhash: key.thumbhash)),
|
|
informationCollector: () => <DiagnosticsNode>[
|
|
DiagnosticsProperty<ImageProvider>('Image provider', this),
|
|
DiagnosticsProperty<String>('Asset Id', key.assetId),
|
|
DiagnosticsProperty<bool>('isAnimated', key.isAnimated),
|
|
],
|
|
onLastListenerRemoved: cancel,
|
|
);
|
|
}
|
|
|
|
return OneFramePlaceholderImageStreamCompleter(
|
|
_codec(key, decode),
|
|
initialImage: getInitialImage(
|
|
RemoteImageProvider.thumbnail(assetId: key.assetId, thumbhash: key.thumbhash, edited: key.edited),
|
|
),
|
|
informationCollector: () => <DiagnosticsNode>[
|
|
DiagnosticsProperty<ImageProvider>('Image provider', this),
|
|
DiagnosticsProperty<String>('Asset Id', key.assetId),
|
|
DiagnosticsProperty<bool>('isAnimated', key.isAnimated),
|
|
],
|
|
onLastListenerRemoved: cancel,
|
|
);
|
|
}
|
|
|
|
Stream<ImageInfo> _codec(RemoteFullImageProvider key, ImageDecoderCallback decode) async* {
|
|
yield* initialImageStream();
|
|
|
|
if (isCancelled) {
|
|
return;
|
|
}
|
|
|
|
final previewRequest = request = RemoteImageRequest(
|
|
uri: getThumbnailUrlForRemoteId(
|
|
key.assetId,
|
|
type: AssetMediaSize.preview,
|
|
thumbhash: key.thumbhash,
|
|
edited: key.edited,
|
|
),
|
|
);
|
|
final loadOriginal = assetType == AssetType.image && AppSetting.get(Setting.loadOriginal);
|
|
yield* loadRequest(previewRequest, decode, isFinal: !loadOriginal);
|
|
|
|
if (!loadOriginal) {
|
|
return;
|
|
}
|
|
|
|
if (isCancelled) {
|
|
return;
|
|
}
|
|
|
|
final originalRequest = request = RemoteImageRequest(
|
|
uri: getOriginalUrlForRemoteId(key.assetId, edited: key.edited),
|
|
);
|
|
yield* loadRequest(originalRequest, decode, isFinal: true);
|
|
}
|
|
|
|
Stream<Object> _animatedCodec(RemoteFullImageProvider key, ImageDecoderCallback decode) async* {
|
|
yield* initialImageStream();
|
|
|
|
if (isCancelled) {
|
|
return;
|
|
}
|
|
|
|
final previewRequest = request = RemoteImageRequest(
|
|
uri: getThumbnailUrlForRemoteId(
|
|
key.assetId,
|
|
type: AssetMediaSize.preview,
|
|
thumbhash: key.thumbhash,
|
|
edited: key.edited,
|
|
),
|
|
);
|
|
yield* loadRequest(previewRequest, decode, isFinal: false);
|
|
|
|
if (isCancelled) {
|
|
return;
|
|
}
|
|
|
|
// always try original for animated, since previews don't support animation
|
|
final originalRequest = request = RemoteImageRequest(
|
|
uri: getOriginalUrlForRemoteId(key.assetId, edited: key.edited),
|
|
);
|
|
final codec = await loadCodecRequest(originalRequest, isFinal: true);
|
|
if (codec == null) {
|
|
if (isCancelled) {
|
|
return;
|
|
}
|
|
throw StateError('Failed to load animated codec for asset ${key.assetId}');
|
|
}
|
|
yield codec;
|
|
}
|
|
|
|
@override
|
|
bool operator ==(Object other) {
|
|
if (identical(this, other)) return true;
|
|
if (other is RemoteFullImageProvider) {
|
|
return assetId == other.assetId &&
|
|
thumbhash == other.thumbhash &&
|
|
isAnimated == other.isAnimated &&
|
|
edited == other.edited;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
@override
|
|
int get hashCode => assetId.hashCode ^ thumbhash.hashCode ^ isAnimated.hashCode ^ edited.hashCode;
|
|
}
|