mirror of
https://github.com/BetterSEQTA/BetterSEQTA-Plus.git
synced 2026-06-06 03:34:40 +00:00
feat(ui/settings): lazyload background image and video swatches
This commit is contained in:
@@ -1,10 +1,10 @@
|
|||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { hasEnoughStorageSpace, isIndexedDBSupported, writeData, openDatabase, readAllData, deleteData } from '@/svelte-interface/hooks/BackgroundDataLoader'
|
import { hasEnoughStorageSpace, isIndexedDBSupported, writeData, openDatabase, readAllData, deleteData } from '@/svelte-interface/hooks/BackgroundDataLoader'
|
||||||
import BackgroundUploader from './BackgroundUploader.svelte';
|
import BackgroundUploader from './BackgroundUploader.svelte';
|
||||||
import BackgroundItem from './BackgroundItem.svelte'
|
import BackgroundItem from './BackgroundItem.svelte'
|
||||||
import { onMount } from 'svelte'
|
import { onMount } from 'svelte'
|
||||||
import { loadBackground } from '@/seqta/ui/ImageBackgrounds'
|
import { loadBackground } from '@/seqta/ui/ImageBackgrounds'
|
||||||
|
import { delay } from 'lodash'
|
||||||
|
|
||||||
let { isEditMode, selectNoBackground = $bindable(), selectedBackground = $bindable() } = $props<{ isEditMode: boolean, selectNoBackground: () => void, selectedBackground: string | null }>();
|
let { isEditMode, selectNoBackground = $bindable(), selectedBackground = $bindable() } = $props<{ isEditMode: boolean, selectNoBackground: () => void, selectedBackground: string | null }>();
|
||||||
let backgrounds = $state<{ id: string; type: string; blob: Blob; url?: string }[]>([]);
|
let backgrounds = $state<{ id: string; type: string; blob: Blob; url?: string }[]>([]);
|
||||||
@@ -14,6 +14,10 @@
|
|||||||
let imageBackgrounds = $derived(backgrounds.filter(bg => bg.type === 'image'));
|
let imageBackgrounds = $derived(backgrounds.filter(bg => bg.type === 'image'));
|
||||||
let videoBackgrounds = $derived(backgrounds.filter(bg => bg.type === 'video'));
|
let videoBackgrounds = $derived(backgrounds.filter(bg => bg.type === 'video'));
|
||||||
|
|
||||||
|
let isVisible = $state(false);
|
||||||
|
let observer: IntersectionObserver;
|
||||||
|
let element: HTMLElement;
|
||||||
|
|
||||||
async function getTheme() {
|
async function getTheme() {
|
||||||
return localStorage.getItem('selectedBackground');
|
return localStorage.getItem('selectedBackground');
|
||||||
}
|
}
|
||||||
@@ -61,8 +65,12 @@
|
|||||||
|
|
||||||
await openDatabase();
|
await openDatabase();
|
||||||
const data = await readAllData();
|
const data = await readAllData();
|
||||||
const dataWithUrls = data.map(bg => ({ ...bg, url: URL.createObjectURL(bg.blob) }));
|
|
||||||
backgrounds = dataWithUrls;
|
if (!isVisible) {
|
||||||
|
backgrounds = data;
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
backgrounds = await preloadBackgrounds(data);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (e instanceof Error) {
|
if (e instanceof Error) {
|
||||||
error = e.message;
|
error = e.message;
|
||||||
@@ -74,6 +82,13 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function preloadBackgrounds(data: { id: string; type: string; blob: Blob }[]): Promise<{ id: string; type: string; blob: Blob; url: string }[]> {
|
||||||
|
return data.map(bg => ({
|
||||||
|
...bg,
|
||||||
|
url: URL.createObjectURL(bg.blob)
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
function selectBackground(fileId: string): void {
|
function selectBackground(fileId: string): void {
|
||||||
if (selectedBackground === fileId) {
|
if (selectedBackground === fileId) {
|
||||||
selectNoBackground();
|
selectNoBackground();
|
||||||
@@ -111,27 +126,48 @@
|
|||||||
selectedBackground
|
selectedBackground
|
||||||
});
|
});
|
||||||
|
|
||||||
onMount(async () => {
|
onMount(() => {
|
||||||
await loadBackgrounds();
|
loadBackgrounds();
|
||||||
selectedBackground = await getTheme();
|
|
||||||
|
observer = new IntersectionObserver((entries) => {
|
||||||
|
if (entries[0].isIntersecting) {
|
||||||
|
delay(() => {
|
||||||
|
isVisible = true;
|
||||||
|
loadBackgrounds();
|
||||||
|
}, 100);
|
||||||
|
selectedBackground = getTheme();
|
||||||
|
observer.disconnect();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
observer.observe(element);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
observer.disconnect();
|
||||||
|
};
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="relative px-1 py-2">
|
<div bind:this={element} class="relative px-1 py-2">
|
||||||
<h2 class="pb-2 text-lg font-bold">Background Images</h2>
|
<h2 class="pb-2 text-lg font-bold">Background Images</h2>
|
||||||
<div class="flex flex-wrap gap-4 mb-4">
|
<div class="flex flex-wrap gap-4 mb-4">
|
||||||
{#if !isEditMode}
|
{#if !isEditMode}
|
||||||
<BackgroundUploader on:fileChange={e => handleFileChange(e.detail)} />
|
<BackgroundUploader on:fileChange={e => handleFileChange(e.detail)} />
|
||||||
{/if}
|
{/if}
|
||||||
{#each imageBackgrounds as bg (bg.id)}
|
{#if isVisible}
|
||||||
<BackgroundItem
|
{#each imageBackgrounds as bg (bg.id)}
|
||||||
{bg}
|
<BackgroundItem
|
||||||
isSelected={selectedBackground === bg.id}
|
bg={bg}
|
||||||
isEditMode={isEditMode}
|
isSelected={selectedBackground === bg.id}
|
||||||
onClick={() => selectBackground(bg.id)}
|
isEditMode={isEditMode}
|
||||||
onDelete={() => deleteBackground(bg.id)}
|
onClick={() => selectBackground(bg.id)}
|
||||||
/>
|
onDelete={() => deleteBackground(bg.id)}/>
|
||||||
{/each}
|
{/each}
|
||||||
|
{:else}
|
||||||
|
{#each imageBackgrounds as bg (bg.id)}
|
||||||
|
<div class="w-16 h-16 rounded-xl bg-zinc-100 dark:bg-zinc-900 animate-pulse"></div>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h2 class="py-2 text-lg font-bold">Background Videos</h2>
|
<h2 class="py-2 text-lg font-bold">Background Videos</h2>
|
||||||
@@ -139,14 +175,20 @@
|
|||||||
{#if !isEditMode}
|
{#if !isEditMode}
|
||||||
<BackgroundUploader on:fileChange={e => handleFileChange(e.detail)} />
|
<BackgroundUploader on:fileChange={e => handleFileChange(e.detail)} />
|
||||||
{/if}
|
{/if}
|
||||||
{#each videoBackgrounds as bg (bg.id)}
|
{#if isVisible}
|
||||||
<BackgroundItem
|
{#each videoBackgrounds as bg (bg.id)}
|
||||||
{bg}
|
<BackgroundItem
|
||||||
isSelected={selectedBackground === bg.id}
|
bg={bg}
|
||||||
isEditMode={isEditMode}
|
isSelected={selectedBackground === bg.id}
|
||||||
onClick={() => selectBackground(bg.id)}
|
isEditMode={isEditMode}
|
||||||
onDelete={() => deleteBackground(bg.id)}
|
onClick={() => selectBackground(bg.id)}
|
||||||
/>
|
onDelete={() => deleteBackground(bg.id)}
|
||||||
{/each}
|
/>
|
||||||
|
{/each}
|
||||||
|
{:else}
|
||||||
|
{#each videoBackgrounds as bg (bg.id)}
|
||||||
|
<div class="w-16 h-16 rounded-xl bg-zinc-100 dark:bg-zinc-900 animate-pulse"></div>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
Reference in New Issue
Block a user