mirror of
https://github.com/immich-app/immich.git
synced 2025-08-11 09:16:31 -04:00
buttery hero animations
This commit is contained in:
parent
a934af6971
commit
abb6f671fe
@ -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
|
||||||
|
@ -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),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -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');
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<ImageInfo> _codec(LocalFullImageProvider key) async {
|
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, 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
|
||||||
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
@ -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) {
|
||||||
|
@ -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,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user