mirror of
https://github.com/immich-app/immich.git
synced 2025-06-21 14:31:03 -04:00
* wip: timeline * more segment extensions * added scrubber * refactor: timeline state * more refactors * fix scrubber segments * added remote thumb & thumbhash provider * feat: merged view * scrub / merged asset fixes * rename stuff & add tile indicators * fix local album timeline query * ignore hidden assets during sync * ignore recovered assets during sync * old scrubber * add video indicator * handle groupBy * handle partner inTimeline * show duration * reduce widget nesting in thumb tile * merge main * chore: extend cacheExtent * ignore touch events on scrub label when not visible * scrub label ignore events and hide immediately * auto reload on sync * refactor image providers * throttle db updates --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
106 lines
3.0 KiB
Dart
106 lines
3.0 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
|
import 'package:immich_mobile/presentation/widgets/images/local_thumb_provider.dart';
|
|
import 'package:immich_mobile/presentation/widgets/images/remote_thumb_provider.dart';
|
|
import 'package:immich_mobile/presentation/widgets/images/thumb_hash_provider.dart';
|
|
import 'package:immich_mobile/widgets/asset_grid/thumbnail_placeholder.dart';
|
|
import 'package:immich_mobile/widgets/common/fade_in_placeholder_image.dart';
|
|
import 'package:logging/logging.dart';
|
|
import 'package:octo_image/octo_image.dart';
|
|
|
|
class Thumbnail extends StatelessWidget {
|
|
const Thumbnail({
|
|
required this.asset,
|
|
this.size = const Size.square(256),
|
|
this.fit = BoxFit.cover,
|
|
super.key,
|
|
});
|
|
|
|
final BaseAsset asset;
|
|
final Size size;
|
|
final BoxFit fit;
|
|
|
|
static ImageProvider imageProvider({
|
|
required BaseAsset asset,
|
|
Size size = const Size.square(256),
|
|
}) {
|
|
if (asset is LocalAsset) {
|
|
return LocalThumbProvider(
|
|
asset: asset,
|
|
height: size.height,
|
|
width: size.width,
|
|
);
|
|
}
|
|
|
|
if (asset is Asset) {
|
|
return RemoteThumbProvider(
|
|
assetId: asset.id,
|
|
height: size.height,
|
|
width: size.width,
|
|
);
|
|
}
|
|
|
|
throw ArgumentError("Unsupported asset type: ${asset.runtimeType}");
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final thumbHash = asset is Asset ? (asset as Asset).thumbHash : null;
|
|
final provider = imageProvider(asset: asset, size: size);
|
|
|
|
return OctoImage.fromSet(
|
|
image: provider,
|
|
octoSet: OctoSet(
|
|
placeholderBuilder: _blurHashPlaceholderBuilder(thumbHash, fit: fit),
|
|
errorBuilder: _blurHashErrorBuilder(
|
|
thumbHash,
|
|
provider: provider,
|
|
fit: fit,
|
|
asset: asset,
|
|
),
|
|
),
|
|
fadeOutDuration: const Duration(milliseconds: 100),
|
|
fadeInDuration: Duration.zero,
|
|
width: size.width,
|
|
height: size.height,
|
|
fit: fit,
|
|
placeholderFadeInDuration: Duration.zero,
|
|
);
|
|
}
|
|
}
|
|
|
|
OctoPlaceholderBuilder _blurHashPlaceholderBuilder(
|
|
String? thumbHash, {
|
|
BoxFit? fit,
|
|
}) {
|
|
return (context) => thumbHash == null
|
|
? const ThumbnailPlaceholder()
|
|
: FadeInPlaceholderImage(
|
|
placeholder: const ThumbnailPlaceholder(),
|
|
image: ThumbHashProvider(thumbHash: thumbHash),
|
|
fit: fit ?? BoxFit.cover,
|
|
);
|
|
}
|
|
|
|
OctoErrorBuilder _blurHashErrorBuilder(
|
|
String? blurhash, {
|
|
BaseAsset? asset,
|
|
ImageProvider? provider,
|
|
BoxFit? fit,
|
|
}) =>
|
|
(context, e, s) {
|
|
Logger("ImThumbnail")
|
|
.warning("Error loading thumbnail for ${asset?.name}", e, s);
|
|
provider?.evict();
|
|
return Stack(
|
|
alignment: Alignment.center,
|
|
children: [
|
|
_blurHashPlaceholderBuilder(blurhash, fit: fit)(context),
|
|
const Opacity(
|
|
opacity: 0.75,
|
|
child: Icon(Icons.error_outline_rounded),
|
|
),
|
|
],
|
|
);
|
|
};
|