Files
OnePieceDle/src/lib/components/ProfileButton.svelte
whidix b849e6c4dc feat: add daily win recording endpoint, user authentication, and profile management
- Implemented a POST endpoint for recording daily wins in the game.
- Created login and signup functionality with email and password.
- Developed a profile page allowing users to update their profile information, change passwords, and manage active sessions.
- Added a toggle feature for switching between login and signup forms.
- Enhanced the layout by removing the profile button and adjusting the header structure.
2026-03-01 23:01:44 +01:00

112 lines
3.0 KiB
Svelte

<script lang="ts">
import { onMount } from 'svelte';
import type { User } from 'better-auth/types';
interface Props {
user: (User & { isAdmin?: boolean }) | null;
}
let { user }: Props = $props();
let isMenuOpen = $state(false);
let menuElement: HTMLDivElement | undefined;
const toggleMenu = () => {
isMenuOpen = !isMenuOpen;
};
const closeMenu = () => {
isMenuOpen = false;
};
const handleLogout = async () => {
const formData = new FormData();
const response = await fetch('/login?/logout', {
method: 'POST',
body: formData
});
if (response.ok) {
window.location.href = '/';
}
};
onMount(() => {
const handleClickOutside = (event: MouseEvent) => {
if (menuElement && !menuElement.contains(event.target as Node)) {
closeMenu();
}
};
document.addEventListener('click', handleClickOutside);
return () => document.removeEventListener('click', handleClickOutside);
});
</script>
<div bind:this={menuElement} class="relative">
{#if user}
<button
onclick={toggleMenu}
class="flex items-center gap-3 rounded-full border border-white/10 bg-white/5 px-2 py-2 pr-4 transition hover:border-amber-300/50 hover:bg-white/10"
>
{#if user.image}
<img
src={user.image}
alt={user.name || 'Profil'}
class="h-8 w-8 rounded-full object-cover"
/>
{:else}
<div class="flex h-8 w-8 items-center justify-center rounded-full bg-amber-300/20 text-xs font-semibold text-amber-100">
{user.name?.charAt(0).toUpperCase() || 'U'}
</div>
{/if}
<span class="max-w-[150px] truncate text-sm font-semibold text-slate-100">
{user.name || 'Utilisateur'}
</span>
<svg
class="h-4 w-4 transition {isMenuOpen ? 'rotate-180' : ''}"
fill="none"
stroke="white"
viewBox="0 0 24 24"
>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M19 14l-7 7m0 0l-7-7m7 7V3" />
</svg>
</button>
{#if isMenuOpen}
<div
class="absolute right-0 top-full mt-2 w-48 rounded-xl border border-white/10 bg-slate-900/95 shadow-[0_24px_60px_rgba(0,0,0,0.45)] backdrop-blur"
>
<a
href="/profile"
onclick={closeMenu}
class="block border-b border-white/5 px-4 py-3 text-sm font-semibold text-slate-100 transition hover:bg-white/5 hover:text-amber-100 first:rounded-t-xl"
>
Voir mon profil
</a>
{#if (user as any).isAdmin}
<a
href="/admin"
onclick={closeMenu}
class="block border-b border-white/5 px-4 py-3 text-sm font-semibold text-amber-300 transition hover:bg-white/5 hover:text-amber-200"
>
Admin
</a>
{/if}
<button
onclick={handleLogout}
class="w-full border-t border-white/5 px-4 py-3 text-sm font-semibold text-red-300 transition hover:bg-red-900/20 last:rounded-b-xl"
>
Se déconnecter
</button>
</div>
{/if}
{:else}
<a
href="/login"
class="rounded-full bg-amber-300 px-5 py-2.5 text-sm font-semibold text-slate-900 transition hover:bg-amber-200"
>
Se connecter
</a>
{/if}
</div>