OPFS에 캐시된 썸네일을 모두 삭제하는 기능 추가

This commit is contained in:
static
2025-07-07 00:30:38 +09:00
parent 8fefbc1bcb
commit e4cce6b8a0
8 changed files with 91 additions and 39 deletions

View File

@@ -1,7 +1,15 @@
<script lang="ts"> <script lang="ts">
let { children } = $props(); import type { Snippet } from "svelte";
import type { ClassValue } from "svelte/elements";
interface Props {
children: Snippet;
class?: ClassValue;
}
let { children, class: className }: Props = $props();
</script> </script>
<div class="flex flex-grow flex-col justify-between px-4"> <div class={["flex flex-grow flex-col justify-between px-4", className]}>
{@render children()} {@render children()}
</div> </div>

View File

@@ -1,5 +1,5 @@
import { LRUCache } from "lru-cache"; import { LRUCache } from "lru-cache";
import { readFile, writeFile, deleteFile } from "$lib/modules/opfs"; import { readFile, writeFile, deleteFile, deleteDirectory } from "$lib/modules/opfs";
import { getThumbnailUrl } from "$lib/modules/thumbnail"; import { getThumbnailUrl } from "$lib/modules/thumbnail";
const loadedThumbnails = new LRUCache<number, string>({ max: 100 }); const loadedThumbnails = new LRUCache<number, string>({ max: 100 });
@@ -27,3 +27,8 @@ export const deleteFileThumbnail = async (fileId: number) => {
loadedThumbnails.delete(fileId); loadedThumbnails.delete(fileId);
await deleteFile(`/thumbnails/${fileId}`); await deleteFile(`/thumbnails/${fileId}`);
}; };
export const deleteAllFileThumbnails = async () => {
loadedThumbnails.clear();
await deleteDirectory("/thumbnails");
};

View File

@@ -59,3 +59,39 @@ export const deleteFile = async (path: string) => {
await parentHandle.removeEntry(filename); await parentHandle.removeEntry(filename);
}; };
const getDirectoryHandle = async (path: string) => {
if (!rootHandle) {
throw new Error("OPFS not prepared");
} else if (path[0] !== "/") {
throw new Error("Path must be absolute");
}
const parts = path.split("/");
if (parts.length <= 1) {
throw new Error("Invalid path");
}
try {
let directoryHandle = rootHandle;
let parentHandle;
for (const part of parts.slice(1)) {
if (!part) continue;
parentHandle = directoryHandle;
directoryHandle = await directoryHandle.getDirectoryHandle(part);
}
return { directoryHandle, parentHandle };
} catch (e) {
if (e instanceof DOMException && e.name === "NotFoundError") {
return {};
}
throw e;
}
};
export const deleteDirectory = async (path: string) => {
const { directoryHandle, parentHandle } = await getDirectoryHandle(path);
if (!parentHandle) return;
await parentHandle.removeEntry(directoryHandle.name, { recursive: true });
};

View File

@@ -4,12 +4,11 @@
import { FullscreenDiv } from "$lib/components/atoms"; import { FullscreenDiv } from "$lib/components/atoms";
import { TopBar } from "$lib/components/molecules"; import { TopBar } from "$lib/components/molecules";
import type { FileCacheIndex } from "$lib/indexedDB"; import type { FileCacheIndex } from "$lib/indexedDB";
import { getFileCacheIndex } from "$lib/modules/file"; import { getFileCacheIndex, deleteFileCache as doDeleteFileCache } from "$lib/modules/file";
import { getFileInfo, type FileInfo } from "$lib/modules/filesystem"; import { getFileInfo, type FileInfo } from "$lib/modules/filesystem";
import { formatFileSize } from "$lib/modules/util"; import { formatFileSize } from "$lib/modules/util";
import { masterKeyStore } from "$lib/stores"; import { masterKeyStore } from "$lib/stores";
import File from "./File.svelte"; import File from "./File.svelte";
import { deleteFileCache as doDeleteFileCache } from "./service";
interface FileCache { interface FileCache {
index: FileCacheIndex; index: FileCacheIndex;

View File

@@ -1,5 +0,0 @@
import { deleteFileCache as doDeleteFileCache } from "$lib/modules/file";
export const deleteFileCache = async (fileId: number) => {
await doDeleteFileCache(fileId);
};

View File

@@ -3,23 +3,26 @@
import { get } from "svelte/store"; import { get } from "svelte/store";
import { goto } from "$app/navigation"; import { goto } from "$app/navigation";
import { BottomDiv, Button, FullscreenDiv } from "$lib/components/atoms"; import { BottomDiv, Button, FullscreenDiv } from "$lib/components/atoms";
import { TopBar } from "$lib/components/molecules"; import { IconEntryButton, TopBar } from "$lib/components/molecules";
import { deleteAllFileThumbnails } from "$lib/modules/file";
import { getFileInfo } from "$lib/modules/filesystem"; import { getFileInfo } from "$lib/modules/filesystem";
import { masterKeyStore } from "$lib/stores"; import { masterKeyStore } from "$lib/stores";
import File from "./File.svelte"; import File from "./File.svelte";
import { import {
persistentStates, persistentStates,
getGenerationStatus, getGenerationStatus,
requestFileThumbnailGeneration, requestThumbnailGeneration,
} from "./service.svelte"; } from "./service.svelte";
import IconDelete from "~icons/material-symbols/delete";
let { data } = $props(); let { data } = $props();
const generateAllThumbnails = async () => { const generateAllThumbnails = () => {
persistentStates.files.forEach(({ info }) => { persistentStates.files.forEach(({ info }) => {
const fileInfo = get(info); const fileInfo = get(info);
if (fileInfo) { if (fileInfo) {
requestFileThumbnailGeneration(fileInfo); requestThumbnailGeneration(fileInfo);
} }
}); });
}; };
@@ -38,31 +41,37 @@
</svelte:head> </svelte:head>
<TopBar title="썸네일" /> <TopBar title="썸네일" />
<FullscreenDiv> <FullscreenDiv class="bg-gray-100 !px-0">
{#if persistentStates.files.length > 0} <div class="flex flex-grow flex-col space-y-4">
<div class="space-y-4 pb-4"> <div class="flex-shrink-0 bg-white p-4 !pt-0">
<div class="space-y-1 break-keep text-gray-800"> <IconEntryButton icon={IconDelete} onclick={deleteAllFileThumbnails} class="w-full">
<p> 저장된 썸네일 모두 삭제하기
{persistentStates.files.length}개 파일의 썸네일이 존재하지 않아요. </IconEntryButton>
</p>
</div>
<div class="space-y-2">
{#each persistentStates.files as { info, status }}
<File
{info}
generationStatus={status}
onclick={({ id }) => goto(`/file/${id}`)}
onGenerateThumbnailClick={requestFileThumbnailGeneration}
/>
{/each}
</div>
</div> </div>
<BottomDiv class="flex flex-col items-center gap-y-2"> {#if persistentStates.files.length > 0}
<div class="flex-grow space-y-2 bg-white p-4">
<p class="text-lg font-bold text-gray-800">썸네일이 누락된 파일</p>
<div class="space-y-4">
<p class="break-keep text-gray-800">
{persistentStates.files.length}개 파일의 썸네일이 존재하지 않아요.
</p>
<div class="space-y-2">
{#each persistentStates.files as { info, status }}
<File
{info}
generationStatus={status}
onclick={({ id }) => goto(`/file/${id}`)}
onGenerateThumbnailClick={requestThumbnailGeneration}
/>
{/each}
</div>
</div>
</div>
{/if}
</div>
{#if persistentStates.files.length > 0}
<BottomDiv class="flex flex-col items-center gap-y-2 px-4">
<Button onclick={generateAllThumbnails} class="w-full">모두 썸네일 생성하기</Button> <Button onclick={generateAllThumbnails} class="w-full">모두 썸네일 생성하기</Button>
</BottomDiv> </BottomDiv>
{:else}
<div class="flex flex-grow items-center justify-center">
<p class="text-gray-500">모든 파일의 썸네일이 존재해요.</p>
</div>
{/if} {/if}
</FullscreenDiv> </FullscreenDiv>

View File

@@ -32,7 +32,7 @@
<ActionEntryButton <ActionEntryButton
class="h-14" class="h-14"
onclick={() => onclick($info)} onclick={() => onclick($info)}
actionButtonIcon={IconCamera} actionButtonIcon={!$generationStatus || $generationStatus === "error" ? IconCamera : undefined}
onActionButtonClick={() => onGenerateThumbnailClick($info)} onActionButtonClick={() => onGenerateThumbnailClick($info)}
actionButtonClass="text-gray-800" actionButtonClass="text-gray-800"
> >

View File

@@ -110,7 +110,7 @@ const requestThumbnailUpload = limitFunction(
{ concurrency: 4 }, { concurrency: 4 },
); );
export const requestFileThumbnailGeneration = async (fileInfo: FileInfo) => { export const requestThumbnailGeneration = async (fileInfo: FileInfo) => {
let status = workingFiles.get(fileInfo.id); let status = workingFiles.get(fileInfo.id);
if (status && get(status) !== "error") return; if (status && get(status) !== "error") return;