ledgerrz/resources/js/components/PasskeyItem.vue
2026-06-15 00:30:57 +02:00

96 lines
3.3 KiB
Vue

<script setup lang="ts">
import { KeyRound, Trash2 } from '@lucide/vue';
import { ref } from 'vue';
import { Button } from '@/components/ui/button';
import {
Dialog,
DialogClose,
DialogContent,
DialogDescription,
DialogFooter,
DialogTitle,
DialogTrigger,
} from '@/components/ui/dialog';
import type { Passkey } from '@/types/auth';
const props = defineProps<{
passkey: Passkey;
}>();
const emit = defineEmits<{
remove: [id: number, onError: () => void];
}>();
const isDeleting = ref(false);
const handleDelete = () => {
isDeleting.value = true;
emit('remove', props.passkey.id, () => {
isDeleting.value = false;
});
};
</script>
<template>
<div class="flex items-center justify-between border-b p-4 last:border-b-0">
<div class="flex items-center gap-4">
<div
class="flex h-10 w-10 shrink-0 items-center justify-center rounded-xl bg-muted"
>
<KeyRound class="h-5 w-5 text-muted-foreground" />
</div>
<div class="space-y-1">
<div class="flex items-center gap-2.5">
<p class="font-medium tracking-tight">{{ passkey.name }}</p>
<span
v-if="passkey.authenticator"
class="inline-flex items-center gap-1 rounded-md bg-muted px-2 py-0.5 text-[11px] font-medium tracking-wide text-muted-foreground uppercase ring-1 ring-border ring-inset"
>
{{ passkey.authenticator }}
</span>
</div>
<p class="text-sm text-muted-foreground">
Added {{ passkey.created_at_diff }}
<template v-if="passkey.last_used_at_diff">
<span class="mx-1 text-muted-foreground/50">/</span>
Last used {{ passkey.last_used_at_diff }}
</template>
</p>
</div>
</div>
<Dialog>
<DialogTrigger as-child>
<Button
variant="ghost"
size="sm"
class="text-destructive hover:bg-destructive/10 hover:text-destructive"
>
<Trash2 class="h-4 w-4" />
<span class="sr-only">Remove</span>
</Button>
</DialogTrigger>
<DialogContent>
<DialogTitle>Remove passkey</DialogTitle>
<DialogDescription>
Are you sure you want to remove the "{{ passkey.name }}"
passkey? You will no longer be able to use it to sign in.
</DialogDescription>
<DialogFooter class="gap-2">
<DialogClose as-child>
<Button variant="secondary">Cancel</Button>
</DialogClose>
<Button
variant="destructive"
:disabled="isDeleting"
@click="handleDelete"
>
{{ isDeleting ? 'Removing...' : 'Remove passkey' }}
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</div>
</template>