mirror of
https://github.com/immich-app/immich.git
synced 2025-06-23 15:30:51 -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>
97 lines
2.9 KiB
Dart
97 lines
2.9 KiB
Dart
import 'dart:async';
|
|
import 'dart:ui';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/widgets.dart';
|
|
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
|
import 'package:immich_mobile/domain/interfaces/asset_media.interface.dart';
|
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
|
import 'package:immich_mobile/infrastructure/repositories/asset_media.repository.dart';
|
|
import 'package:immich_mobile/presentation/widgets/timeline/constants.dart';
|
|
import 'package:immich_mobile/providers/image/cache/thumbnail_image_cache_manager.dart';
|
|
|
|
class LocalThumbProvider extends ImageProvider<LocalThumbProvider> {
|
|
final IAssetMediaRepository _assetMediaRepository =
|
|
const AssetMediaRepository();
|
|
final CacheManager? cacheManager;
|
|
|
|
final LocalAsset asset;
|
|
final double height;
|
|
final double width;
|
|
|
|
LocalThumbProvider({
|
|
required this.asset,
|
|
this.height = kTimelineFixedTileExtent,
|
|
this.width = kTimelineFixedTileExtent,
|
|
this.cacheManager,
|
|
});
|
|
|
|
@override
|
|
Future<LocalThumbProvider> obtainKey(
|
|
ImageConfiguration configuration,
|
|
) {
|
|
return SynchronousFuture(this);
|
|
}
|
|
|
|
@override
|
|
ImageStreamCompleter loadImage(
|
|
LocalThumbProvider key,
|
|
ImageDecoderCallback decode,
|
|
) {
|
|
final cache = cacheManager ?? ThumbnailImageCacheManager();
|
|
return MultiFrameImageStreamCompleter(
|
|
codec: _codec(key, cache, decode),
|
|
scale: 1.0,
|
|
informationCollector: () => <DiagnosticsNode>[
|
|
DiagnosticsProperty<ImageProvider>('Image provider', this),
|
|
DiagnosticsProperty<LocalAsset>('Asset', key.asset),
|
|
],
|
|
);
|
|
}
|
|
|
|
Future<Codec> _codec(
|
|
LocalThumbProvider key,
|
|
CacheManager cache,
|
|
ImageDecoderCallback decode,
|
|
) async {
|
|
final cacheKey = '${key.asset.id}-${key.asset.updatedAt}-${width}x$height';
|
|
|
|
final fileFromCache = await cache.getFileFromCache(cacheKey);
|
|
if (fileFromCache != null) {
|
|
try {
|
|
final buffer =
|
|
await ImmutableBuffer.fromFilePath(fileFromCache.file.path);
|
|
return await decode(buffer);
|
|
} catch (_) {}
|
|
}
|
|
|
|
final thumbnailBytes = await _assetMediaRepository.getThumbnail(
|
|
key.asset.id,
|
|
size: Size(key.width, key.height),
|
|
);
|
|
if (thumbnailBytes == null) {
|
|
PaintingBinding.instance.imageCache.evict(key);
|
|
throw StateError(
|
|
"Loading thumb for local photo ${key.asset.name} failed",
|
|
);
|
|
}
|
|
|
|
final buffer = await ImmutableBuffer.fromUint8List(thumbnailBytes);
|
|
unawaited(cache.putFile(cacheKey, thumbnailBytes));
|
|
return decode(buffer);
|
|
}
|
|
|
|
@override
|
|
bool operator ==(Object other) {
|
|
if (identical(this, other)) return true;
|
|
if (other is LocalThumbProvider) {
|
|
return asset.id == other.asset.id &&
|
|
asset.updatedAt == other.asset.updatedAt;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@override
|
|
int get hashCode => asset.id.hashCode ^ asset.updatedAt.hashCode;
|
|
}
|