Files
pocket-id/frontend/src/lib/components/one-time-link-modal.svelte

87 lines
2.6 KiB
Svelte
Raw Normal View History

2024-08-12 11:00:25 +02:00
<script lang="ts">
import { page } from '$app/state';
import { Button } from '$lib/components/ui/button';
2024-08-12 11:00:25 +02:00
import * as Dialog from '$lib/components/ui/dialog';
import Input from '$lib/components/ui/input/input.svelte';
import Label from '$lib/components/ui/label/label.svelte';
import * as Select from '$lib/components/ui/select/index.js';
import { m } from '$lib/paraglide/messages';
import UserService from '$lib/services/user-service';
import { axiosErrorToast } from '$lib/utils/error-util';
2024-08-12 11:00:25 +02:00
let {
userId = $bindable()
2024-08-12 11:00:25 +02:00
}: {
userId: string | null;
2024-08-12 11:00:25 +02:00
} = $props();
const userService = new UserService();
let oneTimeLink: string | null = $state(null);
let selectedExpiration: keyof typeof availableExpirations = $state(m.one_hour());
let availableExpirations = {
[m.one_hour()]: 60 * 60,
[m.twelve_hours()]: 60 * 60 * 12,
[m.one_day()]: 60 * 60 * 24,
[m.one_week()]: 60 * 60 * 24 * 7,
[m.one_month()]: 60 * 60 * 24 * 30
};
async function createOneTimeAccessToken() {
try {
const expiration = new Date(Date.now() + availableExpirations[selectedExpiration] * 1000);
const token = await userService.createOneTimeAccessToken(expiration, userId!);
oneTimeLink = `${page.url.origin}/lc/${token}`;
} catch (e) {
axiosErrorToast(e);
}
}
2024-08-12 11:00:25 +02:00
function onOpenChange(open: boolean) {
if (!open) {
oneTimeLink = null;
userId = null;
2024-08-12 11:00:25 +02:00
}
}
</script>
<Dialog.Root open={!!userId} {onOpenChange}>
2024-08-12 11:00:25 +02:00
<Dialog.Content class="max-w-md">
<Dialog.Header>
<Dialog.Title>{m.login_code()}</Dialog.Title>
2024-08-12 11:00:25 +02:00
<Dialog.Description
>{m.create_a_login_code_to_sign_in_without_a_passkey_once()}</Dialog.Description
2024-08-12 11:00:25 +02:00
>
</Dialog.Header>
{#if oneTimeLink === null}
<div>
<Label for="expiration">{m.expiration()}</Label>
<Select.Root
selected={{
label: Object.keys(availableExpirations)[0],
value: Object.keys(availableExpirations)[0]
}}
onSelectedChange={(v) =>
(selectedExpiration = v!.value as keyof typeof availableExpirations)}
>
<Select.Trigger class="h-9 ">
<Select.Value>{selectedExpiration}</Select.Value>
</Select.Trigger>
<Select.Content>
{#each Object.keys(availableExpirations) as key}
<Select.Item value={key}>{key}</Select.Item>
{/each}
</Select.Content>
</Select.Root>
</div>
2025-01-19 15:41:16 +01:00
<Button onclick={() => createOneTimeAccessToken()} disabled={!selectedExpiration}>
{m.generate_code()}
</Button>
{:else}
<Label for="login-code" class="sr-only">{m.login_code()}</Label>
<Input id="login-code" value={oneTimeLink} readonly />
{/if}
2024-08-12 11:00:25 +02:00
</Dialog.Content>
</Dialog.Root>