buttery hero animations

This commit is contained in:
mertalev 2025-08-01 18:35:40 -04:00
parent a934af6971
commit abb6f671fe
No known key found for this signature in database
GPG Key ID: DF6ABC77AAD98C95
7 changed files with 163 additions and 15 deletions

View File

@ -1,3 +1,5 @@
import 'dart:ui';
const int noDbId = -9223372036854775808; // from Isar const int noDbId = -9223372036854775808; // from Isar
const double downloadCompleted = -1; const double downloadCompleted = -1;
const double downloadFailed = -2; const double downloadFailed = -2;
@ -28,8 +30,8 @@ const String kDownloadGroupLivePhoto = 'group_livephoto';
const int kTimelineNoneSegmentSize = 120; const int kTimelineNoneSegmentSize = 120;
const int kTimelineAssetLoadBatchSize = 1024; const int kTimelineAssetLoadBatchSize = 1024;
const int kTimelineAssetLoadOppositeSize = 64; const int kTimelineAssetLoadOppositeSize = 64;
const double kTimelineThumbnailTileSize = 256.0; const Size kTimelineThumbnailTileSize = Size.square(256.0);
const double kTimelineThumbnailSize = 384.0; const Size kTimelineThumbnailSize = Size.square(384.0);
const int kTimelineImageCacheMemory = 250 * 1024 * 1024; const int kTimelineImageCacheMemory = 250 * 1024 * 1024;
// Widget keys // Widget keys

View File

@ -473,10 +473,8 @@ class _AssetViewerState extends ConsumerState<AssetViewer> {
if (stackChildren != null && stackChildren.isNotEmpty) { if (stackChildren != null && stackChildren.isNotEmpty) {
asset = stackChildren.elementAt(ref.read(assetViewerProvider.select((s) => s.stackIndex))); asset = stackChildren.elementAt(ref.read(assetViewerProvider.select((s) => s.stackIndex)));
} }
return Container( return Hero(
width: double.infinity, tag: '${asset.heroTag}_$heroOffset',
height: double.infinity,
color: backgroundColor,
child: Thumbnail.fromBaseAsset(asset: asset, fit: BoxFit.contain), child: Thumbnail.fromBaseAsset(asset: asset, fit: BoxFit.contain),
); );
} }

View File

@ -30,7 +30,7 @@ ImageProvider getFullImageProvider(BaseAsset asset, {Size size = const Size(1080
ImageProvider getThumbnailImageProvider({ ImageProvider getThumbnailImageProvider({
BaseAsset? asset, BaseAsset? asset,
String? remoteId, String? remoteId,
Size size = const Size.square(kTimelineThumbnailSize), Size size = kTimelineThumbnailSize,
}) { }) {
assert(asset != null || remoteId != null, 'Either asset or remoteId must be provided'); assert(asset != null || remoteId != null, 'Either asset or remoteId must be provided');

View File

@ -3,7 +3,9 @@ import 'dart:ui';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
import 'package:immich_mobile/constants/constants.dart';
import 'package:immich_mobile/infrastructure/repositories/asset_media.repository.dart'; import 'package:immich_mobile/infrastructure/repositories/asset_media.repository.dart';
import 'package:immich_mobile/presentation/widgets/images/one_frame_multi_image_stream_completer.dart';
class LocalThumbProvider extends ImageProvider<LocalThumbProvider> { class LocalThumbProvider extends ImageProvider<LocalThumbProvider> {
static const _assetMediaRepository = AssetMediaRepository(); static const _assetMediaRepository = AssetMediaRepository();
@ -11,7 +13,7 @@ class LocalThumbProvider extends ImageProvider<LocalThumbProvider> {
final String id; final String id;
final Size size; final Size size;
const LocalThumbProvider({required this.id, required this.size}); const LocalThumbProvider({required this.id, this.size = kTimelineThumbnailSize});
@override @override
Future<LocalThumbProvider> obtainKey(ImageConfiguration configuration) { Future<LocalThumbProvider> obtainKey(ImageConfiguration configuration) {
@ -63,16 +65,45 @@ class LocalFullImageProvider extends ImageProvider<LocalFullImageProvider> {
@override @override
ImageStreamCompleter loadImage(LocalFullImageProvider key, ImageDecoderCallback decode) { ImageStreamCompleter loadImage(LocalFullImageProvider key, ImageDecoderCallback decode) {
return OneFrameImageStreamCompleter(_codec(key)); ImageInfo? thumbnail;
final thumbnailProvider = LocalThumbProvider(id: key.id);
final ImageStreamCompleter? stream = PaintingBinding.instance.imageCache.putIfAbsent(
thumbnailProvider,
() => throw Exception(), // don't bother loading the thumbnail if it isn't cacched
);
if (stream != null) {
void listener(ImageInfo info, bool synchronousCall) {
thumbnail = info;
}
try {
stream.addListener(ImageStreamListener(listener));
} finally {
stream.removeListener(ImageStreamListener(listener));
}
}
return OneFramePlaceholderImageStreamCompleter(
_codec(key, decode),
initialImage: thumbnail,
informationCollector: () => <DiagnosticsNode>[
DiagnosticsProperty<ImageProvider>('Image provider', this),
DiagnosticsProperty<String>('Id', key.id),
DiagnosticsProperty<Size>('Size', key.size),
],
);
} }
Future<ImageInfo> _codec(LocalFullImageProvider key) async { Future<ImageInfo> _codec(LocalFullImageProvider key, ImageDecoderCallback decode) async {
final devicePixelRatio = PlatformDispatcher.instance.views.first.devicePixelRatio; final devicePixelRatio = PlatformDispatcher.instance.views.first.devicePixelRatio;
final codec = await _assetMediaRepository.getLocalThumbnail( final codec = await _assetMediaRepository.getLocalThumbnail(
key.id, key.id,
Size(size.width * devicePixelRatio, size.height * devicePixelRatio), Size(size.width * devicePixelRatio, size.height * devicePixelRatio),
); );
return ImageInfo(image: (await codec.getNextFrame()).image, scale: 1.0); final frame = await codec.getNextFrame();
return ImageInfo(image: frame.image, scale: 1.0);
} }
@override @override

View File

@ -0,0 +1,117 @@
// The below code is adapted from cached_network_image package's
// MultiImageStreamCompleter to better suit one-frame image loading.
// In particular, it allows providing an initial image to emit synchronously.
import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:flutter/painting.dart';
/// An ImageStreamCompleter with support for loading multiple images.
class OneFramePlaceholderImageStreamCompleter extends ImageStreamCompleter {
ImageInfo? _initialImage;
/// The constructor to create an OneFramePlaceholderImageStreamCompleter. The [image]
/// should be the primary image to display. The [initialImage] is an optional
/// image that will be emitted synchronously, useful as a thumbnail or placeholder.
OneFramePlaceholderImageStreamCompleter(
Future<ImageInfo> image, {
ImageInfo? initialImage,
InformationCollector? informationCollector,
}) {
_initialImage = initialImage;
image.then<void>(
setImage,
onError: (Object error, StackTrace stack) {
reportError(
context: ErrorDescription('resolving a single-frame image stream'),
exception: error,
stack: stack,
informationCollector: informationCollector,
silent: true,
);
},
);
}
/// We must avoid disposing a completer if it never had a listener, even
/// if all [keepAlive] handles get disposed.
bool __hadAtLeastOneListener = false;
bool __disposed = false;
@override
void addListener(ImageStreamListener listener) {
__hadAtLeastOneListener = true;
final initialImage = _initialImage;
if (initialImage != null) {
try {
listener.onImage(initialImage.clone(), true);
} catch (exception, stack) {
reportError(
context: ErrorDescription('by a synchronously-called image listener'),
exception: exception,
stack: stack,
);
}
}
super.addListener(listener);
}
@override
void removeListener(ImageStreamListener listener) {
super.removeListener(listener);
if (!hasListeners) {
__maybeDispose();
}
}
int __keepAliveHandles = 0;
@override
ImageStreamCompleterHandle keepAlive() {
final delegateHandle = super.keepAlive();
return _OneFramePlaceholderImageStreamCompleterHandle(this, delegateHandle);
}
void __maybeDispose() {
if (!__hadAtLeastOneListener || __disposed || hasListeners || __keepAliveHandles != 0) {
return;
}
__disposed = true;
}
@override
void onDisposed() {
_initialImage?.dispose();
_initialImage = null;
super.onDisposed();
}
}
class _OneFramePlaceholderImageStreamCompleterHandle implements ImageStreamCompleterHandle {
_OneFramePlaceholderImageStreamCompleterHandle(this._completer, this._delegateHandle) {
_completer!.__keepAliveHandles += 1;
}
OneFramePlaceholderImageStreamCompleter? _completer;
final ImageStreamCompleterHandle _delegateHandle;
/// Call this method to signal the [ImageStreamCompleter] that it can now be
/// disposed when its last listener drops.
///
/// This method must only be called once per object.
@override
void dispose() {
assert(_completer != null);
assert(_completer!.__keepAliveHandles > 0);
assert(!_completer!.__disposed);
_delegateHandle.dispose();
_completer!.__keepAliveHandles -= 1;
_completer!.__maybeDispose();
_completer = null;
}
}

View File

@ -29,7 +29,7 @@ class Thumbnail extends StatefulWidget {
const Thumbnail({ const Thumbnail({
this.imageProvider, this.imageProvider,
this.fit = BoxFit.cover, this.fit = BoxFit.cover,
this.size = const ui.Size.square(kTimelineThumbnailSize), this.size = kTimelineThumbnailSize,
this.blurhash, this.blurhash,
this.thumbhashMode = ThumbhashMode.enabled, this.thumbhashMode = ThumbhashMode.enabled,
super.key, super.key,
@ -38,7 +38,7 @@ class Thumbnail extends StatefulWidget {
Thumbnail.fromAsset({ Thumbnail.fromAsset({
required Asset asset, required Asset asset,
this.fit = BoxFit.cover, this.fit = BoxFit.cover,
this.size = const ui.Size.square(kTimelineThumbnailSize), this.size = kTimelineThumbnailSize,
this.thumbhashMode = ThumbhashMode.enabled, this.thumbhashMode = ThumbhashMode.enabled,
super.key, super.key,
}) : blurhash = asset.thumbhash, }) : blurhash = asset.thumbhash,
@ -47,7 +47,7 @@ class Thumbnail extends StatefulWidget {
Thumbnail.fromBaseAsset({ Thumbnail.fromBaseAsset({
required BaseAsset? asset, required BaseAsset? asset,
this.fit = BoxFit.cover, this.fit = BoxFit.cover,
this.size = const ui.Size.square(kTimelineThumbnailSize), this.size = kTimelineThumbnailSize,
this.thumbhashMode = ThumbhashMode.enabled, this.thumbhashMode = ThumbhashMode.enabled,
super.key, super.key,
}) : blurhash = switch (asset) { }) : blurhash = switch (asset) {

View File

@ -13,7 +13,7 @@ import 'package:immich_mobile/providers/timeline/multiselect.provider.dart';
class ThumbnailTile extends ConsumerWidget { class ThumbnailTile extends ConsumerWidget {
const ThumbnailTile( const ThumbnailTile(
this.asset, { this.asset, {
this.size = const Size.square(kTimelineThumbnailTileSize), this.size = kTimelineThumbnailTileSize,
this.fit = BoxFit.cover, this.fit = BoxFit.cover,
this.showStorageIndicator = true, this.showStorageIndicator = true,
this.lockSelection = false, this.lockSelection = false,