mirror of
https://github.com/immich-app/immich.git
synced 2025-12-10 03:23:45 +09:00
Some checks are pending
CodeQL / Analyze (javascript) (push) Waiting to run
CodeQL / Analyze (python) (push) Waiting to run
Docker / pre-job (push) Waiting to run
Docker / Re-Tag ML () (push) Blocked by required conditions
Docker / Re-Tag ML (-armnn) (push) Blocked by required conditions
Docker / Re-Tag ML (-cuda) (push) Blocked by required conditions
Docker / Re-Tag ML (-openvino) (push) Blocked by required conditions
Docker / Re-Tag ML (-rknn) (push) Blocked by required conditions
Docker / Re-Tag ML (-rocm) (push) Blocked by required conditions
Docker / Re-Tag Server () (push) Blocked by required conditions
Docker / Build and Push ML (armnn, linux/arm64, -armnn) (push) Blocked by required conditions
Docker / Build and Push ML (cpu, ) (push) Blocked by required conditions
Docker / Build and Push ML (cuda, linux/amd64, -cuda) (push) Blocked by required conditions
Docker / Build and Push ML (openvino, linux/amd64, -openvino) (push) Blocked by required conditions
Docker / Build and Push ML (rknn, linux/arm64, -rknn) (push) Blocked by required conditions
Docker / Build and Push ML (rocm, linux/amd64, {"linux/amd64": "mich"}, -rocm) (push) Blocked by required conditions
Docker / Build and Push Server (push) Blocked by required conditions
Docker / Docker Build & Push Server Success (push) Blocked by required conditions
Docker / Docker Build & Push ML Success (push) Blocked by required conditions
Docs build / pre-job (push) Waiting to run
Docs build / Docs Build (push) Blocked by required conditions
Static Code Analysis / pre-job (push) Waiting to run
Static Code Analysis / Run Dart Code Analysis (push) Blocked by required conditions
Static Code Analysis / zizmor (push) Waiting to run
Test / pre-job (push) Waiting to run
Test / Test & Lint Server (push) Blocked by required conditions
Test / Unit Test CLI (push) Blocked by required conditions
Test / Unit Test CLI (Windows) (push) Blocked by required conditions
Test / Lint Web (push) Blocked by required conditions
Test / Test Web (push) Blocked by required conditions
Test / Test i18n (push) Blocked by required conditions
Test / End-to-End Lint (push) Blocked by required conditions
Test / Medium Tests (Server) (push) Blocked by required conditions
Test / End-to-End Tests (Server & CLI) (ubuntu-24.04-arm) (push) Blocked by required conditions
Test / End-to-End Tests (Server & CLI) (ubuntu-latest) (push) Blocked by required conditions
Test / End-to-End Tests (Web) (ubuntu-24.04-arm) (push) Blocked by required conditions
Test / End-to-End Tests (Web) (ubuntu-latest) (push) Blocked by required conditions
Test / End-to-End Tests Success (push) Blocked by required conditions
Test / Unit Test Mobile (push) Blocked by required conditions
Test / Unit Test ML (push) Blocked by required conditions
Test / .github Files Formatting (push) Blocked by required conditions
Test / ShellCheck (push) Waiting to run
Test / OpenAPI Clients (push) Waiting to run
Test / SQL Schema Checks (push) Waiting to run
72 lines
1.9 KiB
Svelte
72 lines
1.9 KiB
Svelte
<script lang="ts">
|
|
import LoadingSpinner from '$lib/components/shared-components/loading-spinner.svelte';
|
|
import { assetViewerFadeDuration } from '$lib/constants';
|
|
import type { TimelineAsset } from '$lib/managers/timeline-manager/types';
|
|
import { getAssetThumbnailUrl } from '$lib/utils';
|
|
import { getAltText } from '$lib/utils/thumbnail-util';
|
|
import { AssetMediaSize } from '@immich/sdk';
|
|
import { onMount } from 'svelte';
|
|
import { fade } from 'svelte/transition';
|
|
|
|
interface Props {
|
|
asset: TimelineAsset;
|
|
onImageLoad: () => void;
|
|
}
|
|
|
|
const { asset, onImageLoad }: Props = $props();
|
|
|
|
let assetFileUrl: string = $state('');
|
|
let imageLoaded: boolean = $state(false);
|
|
let loader = $state<HTMLImageElement>();
|
|
|
|
const onLoadCallback = () => {
|
|
imageLoaded = true;
|
|
assetFileUrl = imageLoaderUrl;
|
|
onImageLoad();
|
|
};
|
|
|
|
onMount(() => {
|
|
if (loader?.complete) {
|
|
onLoadCallback();
|
|
}
|
|
loader?.addEventListener('load', onLoadCallback);
|
|
return () => {
|
|
loader?.removeEventListener('load', onLoadCallback);
|
|
};
|
|
});
|
|
|
|
const imageLoaderUrl = $derived(getAssetThumbnailUrl({ id: asset.id, size: AssetMediaSize.Preview }));
|
|
</script>
|
|
|
|
{#if !imageLoaded}
|
|
<!-- svelte-ignore a11y_missing_attribute -->
|
|
<img bind:this={loader} style="display:none" src={imageLoaderUrl} aria-hidden="true" />
|
|
{/if}
|
|
|
|
{#if !imageLoaded}
|
|
<div id="spinner" class="flex h-full items-center justify-center">
|
|
<LoadingSpinner />
|
|
</div>
|
|
{:else if imageLoaded}
|
|
<div transition:fade={{ duration: assetViewerFadeDuration }} class="h-full w-full">
|
|
<img
|
|
class="h-full w-full rounded-2xl object-contain transition-all"
|
|
src={assetFileUrl}
|
|
alt={$getAltText(asset)}
|
|
draggable="false"
|
|
/>
|
|
</div>
|
|
{/if}
|
|
|
|
<style>
|
|
@keyframes delayedVisibility {
|
|
to {
|
|
visibility: visible;
|
|
}
|
|
}
|
|
#spinner {
|
|
visibility: hidden;
|
|
animation: 0s linear 0.4s forwards delayedVisibility;
|
|
}
|
|
</style>
|