DirectroyEntry 컴포넌트 구현

This commit is contained in:
static
2025-01-04 02:46:19 +09:00
parent da18e6856a
commit 5115217153
10 changed files with 109 additions and 31 deletions

View File

@@ -5,14 +5,14 @@
interface Props { interface Props {
children: Snippet; children: Snippet;
onClose?: () => void; onclose?: () => void;
isOpen: boolean; isOpen: boolean;
} }
let { children, onClose, isOpen = $bindable() }: Props = $props(); let { children, onclose, isOpen = $bindable() }: Props = $props();
const closeModal = $derived( const closeModal = $derived(
onClose || onclose ||
(() => { (() => {
isOpen = false; isOpen = false;
}), }),

View File

@@ -0,0 +1,29 @@
<script lang="ts">
import type { Snippet } from "svelte";
import IconArrowBack from "~icons/material-symbols/arrow-back";
interface Props {
children?: Snippet;
onback?: () => void;
title?: string;
}
let { children, onback, title }: Props = $props();
const back = $derived(() => {
setTimeout(onback || (() => history.back()), 100);
});
</script>
<div class="relative mt-4 flex items-center justify-between">
<button onclick={back} class="rounded-full p-1 active:bg-gray-100">
<IconArrowBack class="text-2xl" />
</button>
{#if title}
<p class="absolute left-1/2 -translate-x-1/2 transform text-xl font-semibold">{title}</p>
{/if}
{#if children}
{@render children?.()}
{/if}
</div>

View File

@@ -1,2 +1,3 @@
export { default as BottomSheet } from "./BottomSheet.svelte"; export { default as BottomSheet } from "./BottomSheet.svelte";
export { default as Modal } from "./Modal.svelte"; export { default as Modal } from "./Modal.svelte";
export { default as TopBar } from "./TopBar.svelte";

View File

@@ -46,7 +46,7 @@
onMount(async () => { onMount(async () => {
const res = await refreshToken(); const res = await refreshToken();
if (res.ok) { if (res.ok) {
await goto(data.redirectPath); await goto(data.redirectPath, { replaceState: true });
} }
}); });
</script> </script>

View File

@@ -21,7 +21,7 @@
($clientKeyStore && ($clientKeyStore &&
(await requestMasterKeyDownload($clientKeyStore.decryptKey, $clientKeyStore.verifyKey))) (await requestMasterKeyDownload($clientKeyStore.decryptKey, $clientKeyStore.verifyKey)))
) { ) {
await goto(data.redirectPath); await goto(data.redirectPath, { replaceState: true });
} }
}); });
</script> </script>

View File

@@ -47,7 +47,7 @@
onMount(async () => { onMount(async () => {
if ($clientKeyStore) { if ($clientKeyStore) {
await goto(data.redirectPath); await goto(data.redirectPath, { replaceState: true });
} }
}); });
</script> </script>

View File

@@ -21,7 +21,10 @@ export const load: PageServerLoad = async ({ params, fetch }) => {
directoryInfo.subDirectories.map(async (subDirectoryId) => { directoryInfo.subDirectories.map(async (subDirectoryId) => {
const res = await fetch(`/api/directory/${subDirectoryId}`); const res = await fetch(`/api/directory/${subDirectoryId}`);
if (!res.ok) error(500, "Internal server error"); if (!res.ok) error(500, "Internal server error");
return (await res.json()) as DirectroyInfoResponse; return {
...((await res.json()) as DirectroyInfoResponse),
id: subDirectoryId,
};
}), }),
); );
const fileInfos = directoryInfo.files; // TODO const fileInfos = directoryInfo.files; // TODO

View File

@@ -1,4 +1,5 @@
<script lang="ts"> <script lang="ts">
import { TopBar } from "$lib/components";
import { FloatingButton } from "$lib/components/buttons"; import { FloatingButton } from "$lib/components/buttons";
import { clientKeyStore, masterKeyStore } from "$lib/stores"; import { clientKeyStore, masterKeyStore } from "$lib/stores";
import CreateBottomSheet from "./CreateBottomSheet.svelte"; import CreateBottomSheet from "./CreateBottomSheet.svelte";
@@ -20,25 +21,23 @@
return decryptDirectroyMetadata(metadata, $masterKeyStore.get(metadata.mekVersion)!.key); return decryptDirectroyMetadata(metadata, $masterKeyStore.get(metadata.mekVersion)!.key);
} }
}); });
const subDirectoryMetadatas = $derived.by(() => { const subDirectories = $derived.by(() => {
const { subDirectories } = data; const { subDirectories } = data;
if ($masterKeyStore) { if ($masterKeyStore) {
return Promise.all( return Promise.all(
subDirectories.map(async (subDirectory) => { subDirectories.map(async (subDirectory) => {
const metadata = subDirectory.metadata!; const metadata = subDirectory.metadata!;
return await decryptDirectroyMetadata( return {
metadata, ...(await decryptDirectroyMetadata(
$masterKeyStore.get(metadata.mekVersion)!.key, metadata,
); $masterKeyStore.get(metadata.mekVersion)!.key,
)),
id: subDirectory.id,
};
}), }),
); ).then((subDirectories) => {
} subDirectories.sort((a, b) => a.name.localeCompare(b.name));
}); return subDirectories;
const entries = $derived.by(() => {
if (subDirectoryMetadatas) {
return subDirectoryMetadatas.then((subDirectroyMetadatas) => {
subDirectroyMetadatas.sort((a, b) => a.name.localeCompare(b.name));
return subDirectroyMetadatas;
}); });
} }
}); });
@@ -58,12 +57,24 @@
<title>파일</title> <title>파일</title>
</svelte:head> </svelte:head>
<div class="relative h-full"> <div class="relative flex h-full flex-col px-4">
<div> {#if data.id !== "root"}
{#if entries} {#if !metadata}
{#await entries then entries} <TopBar />
{#each entries as { name }} {:else}
<DirectoryEntry {name} /> {#await metadata}
<TopBar />
{:then metadata}
<TopBar title={metadata.name} />
{/await}
{/if}
{/if}
<div class="mt-4">
{#if subDirectories}
{#await subDirectories then subDirectories}
{#each subDirectories as { id, name }}
<DirectoryEntry {id} {name} />
{/each} {/each}
{/await} {/await}
{/if} {/if}

View File

@@ -18,7 +18,7 @@
}; };
</script> </script>
<Modal bind:isOpen onClose={closeModal}> <Modal bind:isOpen onclose={closeModal}>
<div class="flex flex-col px-1"> <div class="flex flex-col px-1">
<p class="text-xl font-bold">새 폴더</p> <p class="text-xl font-bold">새 폴더</p>
<div class="my-4 flex w-full"> <div class="my-4 flex w-full">

View File

@@ -1,12 +1,46 @@
<script lang="ts"> <script lang="ts">
import { goto } from "$app/navigation";
import IconFolder from "~icons/material-symbols/folder";
import IconMoreVert from "~icons/material-symbols/more-vert";
interface Props { interface Props {
id: number;
name: string; name: string;
} }
let { name }: Props = $props(); let { id, name }: Props = $props();
const openDirectory = () => {
setTimeout(() => {
goto(`/directory/${id}`);
}, 100);
};
</script> </script>
<div> <!-- svelte-ignore a11y_no_static_element_interactions -->
<!-- TODO --> <!-- svelte-ignore a11y_click_events_have_key_events -->
<p>{name}</p> <div id="button" onclick={openDirectory} class="h-12 w-full rounded-xl">
<div id="button-content" class="flex h-full items-center justify-between p-2 transition">
<div class="flex items-center gap-x-2">
<IconFolder class="text-lg" />
<p class="font-medium">{name}</p>
</div>
<button
id="open-menu"
onclick={(e) => e.stopPropagation()}
class="rounded-full p-1 active:bg-gray-100"
>
<IconMoreVert class="text-lg transition active:scale-95" />
</button>
</div>
</div> </div>
<style>
#button:active:not(:has(#open-menu:active)) {
@apply bg-gray-100;
}
#button-content:active:not(:has(#open-menu:active)) {
@apply scale-95;
}
</style>