mirror of
https://github.com/immich-app/immich.git
synced 2026-05-18 03:10:24 +03:00
7e2863858d
Replace ImageManager with a new AdaptiveImageLoader that progressively loads images through quality tiers (thumbnail → preview → original). New components and utilities: - AdaptiveImage: layered image renderer with thumbhash, thumbnail, preview, and original layers with visibility managed by load state - AdaptiveImageLoader: state machine driving the quality progression with per-quality callbacks and error handling - ImageLayer/Image: low-level image elements with load/error lifecycle - PreloadManager: preloads adjacent assets for instant navigation - AlphaBackground/DelayedLoadingSpinner: loading state UI Zoom is handled via a derived CSS transform applied to the content wrapper in AdaptiveImage, with the zoom library (zoomTarget: null) only tracking state without manipulating the DOM directly. Also adds scaleToCover to container-utils and getAssetUrls to utils.
40 lines
1.2 KiB
TypeScript
40 lines
1.2 KiB
TypeScript
import { assetViewerManager } from '$lib/managers/asset-viewer-manager.svelte';
|
|
import { createZoomImageWheel } from '@zoom-image/core';
|
|
|
|
export const zoomImageAction = (node: HTMLElement, options?: { disabled?: boolean }) => {
|
|
const zoomInstance = createZoomImageWheel(node, {
|
|
maxZoom: 10,
|
|
initialState: assetViewerManager.zoomState,
|
|
zoomTarget: null,
|
|
});
|
|
|
|
const unsubscribes = [
|
|
assetViewerManager.on({ ZoomChange: (state) => zoomInstance.setState(state) }),
|
|
zoomInstance.subscribe(({ state }) => assetViewerManager.onZoomChange(state)),
|
|
];
|
|
|
|
const stopIfDisabled = (event: Event) => {
|
|
if (options?.disabled) {
|
|
event.stopImmediatePropagation();
|
|
}
|
|
};
|
|
|
|
node.addEventListener('wheel', stopIfDisabled, { capture: true });
|
|
node.addEventListener('pointerdown', stopIfDisabled, { capture: true });
|
|
|
|
node.style.overflow = 'visible';
|
|
return {
|
|
update(newOptions?: { disabled?: boolean }) {
|
|
options = newOptions;
|
|
},
|
|
destroy() {
|
|
for (const unsubscribe of unsubscribes) {
|
|
unsubscribe();
|
|
}
|
|
node.removeEventListener('wheel', stopIfDisabled, { capture: true });
|
|
node.removeEventListener('pointerdown', stopIfDisabled, { capture: true });
|
|
zoomInstance.cleanup();
|
|
},
|
|
};
|
|
};
|