BenutzerProfil

This commit is contained in:
Ashikagi
2026-03-23 21:38:11 +01:00
parent f5338ea3b2
commit cd77f88d96
22 changed files with 1145 additions and 109 deletions

View File

@@ -0,0 +1,369 @@
import React, { useEffect, useRef, useState } from 'react';
import { Check, ChevronDown } from 'lucide-react';
import { countryOptions, getCountryLabel, getCountryOption } from '../data/countries';
import CountryFlag from './CountryFlag';
const emptyStatus = {
message: '',
error: ''
};
export default function ProfileTab({
session,
profile,
loading,
error,
onSave,
onDelete,
onSessionChange
}) {
const [formValues, setFormValues] = useState(profile);
const [status, setStatus] = useState(emptyStatus);
const [saving, setSaving] = useState(false);
const [deleting, setDeleting] = useState(false);
const [isCountryOpen, setIsCountryOpen] = useState(false);
const countryDropdownRef = useRef(null);
const selectedCountry = getCountryOption(formValues.country);
const hasCustomCountryValue = Boolean(formValues.country) && !selectedCountry;
useEffect(() => {
setFormValues(profile);
}, [profile]);
useEffect(() => {
if (!isCountryOpen) return undefined;
const handlePointerDown = (event) => {
if (countryDropdownRef.current && !countryDropdownRef.current.contains(event.target)) {
setIsCountryOpen(false);
}
};
const handleKeyDown = (event) => {
if (event.key === 'Escape') {
setIsCountryOpen(false);
}
};
document.addEventListener('mousedown', handlePointerDown);
document.addEventListener('keydown', handleKeyDown);
return () => {
document.removeEventListener('mousedown', handlePointerDown);
document.removeEventListener('keydown', handleKeyDown);
};
}, [isCountryOpen]);
const updateField = (field) => (event) => {
const nextValue = field === 'testsVisible'
? event.target.checked
: event.target.value;
setFormValues((current) => ({
...current,
[field]: nextValue
}));
};
const handleCountrySelect = (countryCode) => {
setFormValues((current) => ({
...current,
country: countryCode
}));
setIsCountryOpen(false);
};
const handleSubmit = async (event) => {
event.preventDefault();
setSaving(true);
setStatus(emptyStatus);
const result = await onSave(formValues);
if (result?.ok) {
if (result.session) {
onSessionChange?.(result.session);
}
setStatus({
message: result.emailUpdateRequested
? 'Profil gespeichert. Die neue Mailadresse muss ggf. noch per E-Mail bestätigt werden.'
: 'Profil gespeichert.',
error: ''
});
} else {
setStatus({
message: '',
error: result?.error?.message || 'Profil konnte nicht gespeichert werden.'
});
}
setSaving(false);
};
const handleDelete = async () => {
const confirmed = window.confirm(
'Profil wirklich löschen?\n\nDabei werden alle Tests dieses Benutzers sowie vorhandene Analysepläne dauerhaft entfernt.'
);
if (!confirmed) return;
setDeleting(true);
setStatus(emptyStatus);
const result = await onDelete?.();
if (result?.ok) {
setStatus({
message: 'Profil und alle Tests wurden gelöscht. Das Konto bleibt bestehen und kann neu eingerichtet werden.',
error: ''
});
} else {
setStatus({
message: '',
error: result?.error?.message || 'Profil konnte nicht gelöscht werden.'
});
}
setDeleting(false);
};
return (
<div className="min-h-screen bg-gradient-to-br from-slate-50 via-sky-50 to-emerald-100 dark:from-gray-950 dark:via-gray-900 dark:to-gray-950 p-6 text-gray-900 dark:text-gray-100">
<div className="max-w-4xl mx-auto">
<div className="bg-white dark:bg-gray-900 rounded-2xl shadow-lg border border-gray-200 dark:border-gray-800 overflow-hidden">
<div className="px-6 py-5 border-b border-gray-200 dark:border-gray-800">
<p className="text-xs uppercase tracking-[0.25em] text-sky-700 dark:text-sky-300">Benutzerprofil</p>
<h1 className="text-3xl font-bold mt-2">Profil verwalten</h1>
<p className="mt-2 text-sm text-gray-600 dark:text-gray-300">
Name, Vorname, Mailadresse, Land und die Sichtbarkeit deiner Testfreigaben werden hier gepflegt.
</p>
</div>
{loading ? (
<div className="px-6 py-10 text-sm text-gray-600 dark:text-gray-300">
Profil wird geladen...
</div>
) : (
<form onSubmit={handleSubmit} className="px-6 py-6 space-y-6">
<div className="grid md:grid-cols-2 gap-5">
<label className="block">
<span className="text-sm font-semibold text-gray-700 dark:text-gray-200">Vorname</span>
<input
type="text"
value={formValues.firstName || ''}
onChange={updateField('firstName')}
className="mt-1 w-full px-4 py-3 rounded-lg border border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800 focus:outline-none focus:ring-2 focus:ring-sky-500"
placeholder="Max"
/>
</label>
<label className="block">
<span className="text-sm font-semibold text-gray-700 dark:text-gray-200">Nachname</span>
<input
type="text"
value={formValues.lastName || ''}
onChange={updateField('lastName')}
className="mt-1 w-full px-4 py-3 rounded-lg border border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800 focus:outline-none focus:ring-2 focus:ring-sky-500"
placeholder="Mustermann"
/>
</label>
<label className="block md:col-span-2">
<span className="text-sm font-semibold text-gray-700 dark:text-gray-200">Mailadresse</span>
<input
type="email"
value={formValues.email || session?.user?.email || ''}
onChange={updateField('email')}
required
className="mt-1 w-full px-4 py-3 rounded-lg border border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800 focus:outline-none focus:ring-2 focus:ring-sky-500"
placeholder="you@example.com"
/>
<span className="mt-2 block text-xs text-gray-500 dark:text-gray-400">
Wenn du die Mailadresse änderst, wird dir eine Bestätigung an die neue Adresse gesendet.
</span>
</label>
<label className="block">
<span className="text-sm font-semibold text-gray-700 dark:text-gray-200">Land</span>
<div ref={countryDropdownRef} className="relative mt-1">
<button
type="button"
onClick={() => setIsCountryOpen((current) => !current)}
className="flex w-full items-center justify-between rounded-lg border border-gray-200 bg-white px-4 py-3 text-left text-sm text-gray-800 transition hover:border-sky-400 focus:outline-none focus:ring-2 focus:ring-sky-500 dark:border-gray-700 dark:bg-gray-800 dark:text-gray-100"
>
<span className="inline-flex items-center gap-3">
<CountryFlag
countryCode={selectedCountry?.code || formValues.country}
title={selectedCountry?.name || getCountryLabel(formValues.country) || 'Land auswählen'}
className="h-5 w-7 rounded-[3px] shadow-sm ring-1 ring-black/10"
/>
<span className="font-medium">
{selectedCountry?.name || getCountryLabel(formValues.country) || 'Land auswählen'}
</span>
</span>
<ChevronDown
className={`h-4 w-4 text-gray-500 transition ${isCountryOpen ? 'rotate-180' : ''}`}
aria-hidden="true"
/>
</button>
{isCountryOpen && (
<div className="absolute left-0 right-0 z-20 mt-2 max-h-72 overflow-y-auto rounded-xl border border-gray-200 bg-white p-2 shadow-2xl dark:border-gray-700 dark:bg-gray-900">
<button
type="button"
onClick={() => handleCountrySelect('')}
className={`flex w-full items-center justify-between rounded-lg px-3 py-2 text-left text-sm transition ${
!formValues.country
? 'bg-sky-50 text-sky-700 dark:bg-sky-950/40 dark:text-sky-200'
: 'text-gray-700 hover:bg-gray-100 dark:text-gray-200 dark:hover:bg-gray-800'
}`}
>
<span className="inline-flex items-center gap-3">
<CountryFlag
countryCode=""
title="Kein Land ausgewählt"
className="h-5 w-7 rounded-[3px] shadow-sm"
/>
<span>Kein Land ausgewählt</span>
</span>
{!formValues.country && <Check className="h-4 w-4" aria-hidden="true" />}
</button>
{hasCustomCountryValue && (
<button
type="button"
onClick={() => handleCountrySelect(formValues.country)}
className="flex w-full items-center justify-between rounded-lg px-3 py-2 text-left text-sm text-gray-700 transition hover:bg-gray-100 dark:text-gray-200 dark:hover:bg-gray-800"
>
<span className="inline-flex items-center gap-3">
<CountryFlag
countryCode={formValues.country}
title={getCountryLabel(formValues.country)}
className="h-5 w-7 rounded-[3px] shadow-sm ring-1 ring-black/10"
/>
<span>{getCountryLabel(formValues.country)}</span>
</span>
<Check className="h-4 w-4" aria-hidden="true" />
</button>
)}
{countryOptions.map((country) => {
const isSelected = country.code === formValues.country;
return (
<button
key={country.code}
type="button"
onClick={() => handleCountrySelect(country.code)}
className={`flex w-full items-center justify-between rounded-lg px-3 py-2 text-left text-sm transition ${
isSelected
? 'bg-sky-50 text-sky-700 dark:bg-sky-950/40 dark:text-sky-200'
: 'text-gray-700 hover:bg-gray-100 dark:text-gray-200 dark:hover:bg-gray-800'
}`}
>
<span className="inline-flex items-center gap-3">
<CountryFlag
countryCode={country.code}
title={country.name}
className="h-5 w-7 rounded-[3px] shadow-sm ring-1 ring-black/10"
/>
<span>{country.name}</span>
</span>
{isSelected && <Check className="h-4 w-4" aria-hidden="true" />}
</button>
);
})}
</div>
)}
</div>
<span className="mt-2 block text-xs text-gray-500 dark:text-gray-400">
Die ausgewählte Flagge wird später auch im Kopfbereich neben deiner Mailadresse angezeigt.
</span>
</label>
<div className="block">
<span className="text-sm font-semibold text-gray-700 dark:text-gray-200">Ranglistenplatz</span>
<div className="mt-1 rounded-lg border border-dashed border-amber-300 bg-amber-50 px-4 py-3 dark:border-amber-900/70 dark:bg-amber-950/30">
<div className="flex items-center justify-between gap-3">
<span className="text-sm font-medium text-amber-800 dark:text-amber-200">
Platzhalter
</span>
<span className="rounded-full bg-white px-3 py-1 text-sm font-semibold text-amber-700 shadow-sm dark:bg-gray-900 dark:text-amber-300">
# -
</span>
</div>
<p className="mt-2 text-xs text-amber-700/80 dark:text-amber-200/80">
Hier kann spater der Ranglistenplatz des Benutzers angezeigt werden.
</p>
</div>
</div>
</div>
<div className="rounded-xl border border-sky-200 dark:border-sky-900/70 bg-sky-50 dark:bg-sky-950/30 p-5">
<div className="flex items-start justify-between gap-4 flex-wrap">
<div>
<h2 className="text-lg font-semibold text-gray-900 dark:text-gray-100">Tests sichtbar</h2>
<p className="mt-2 text-sm text-gray-600 dark:text-gray-300 max-w-2xl">
Ist dieser Schalter aktiv, funktionieren deine Freigabelinks. Wenn du ihn deaktivierst,
bleiben vorhandene Links gespeichert, sind aber nicht mehr öffentlich sichtbar.
</p>
</div>
<label className="inline-flex items-center gap-3 rounded-full border border-sky-300 dark:border-sky-800 px-4 py-2 bg-white dark:bg-gray-900">
<input
type="checkbox"
checked={Boolean(formValues.testsVisible)}
onChange={updateField('testsVisible')}
className="h-4 w-4 rounded border-gray-300 text-sky-600 focus:ring-sky-500"
/>
<span className="text-sm font-semibold text-gray-800 dark:text-gray-100">
{formValues.testsVisible ? 'Sichtbar' : 'Nicht sichtbar'}
</span>
</label>
</div>
</div>
{(error || status.error) && (
<div className="rounded-lg border border-rose-200 dark:border-rose-900/70 bg-rose-50 dark:bg-rose-950/30 px-4 py-3 text-sm text-rose-700 dark:text-rose-200">
{status.error || error}
</div>
)}
{status.message && (
<div className="rounded-lg border border-emerald-200 dark:border-emerald-900/70 bg-emerald-50 dark:bg-emerald-950/30 px-4 py-3 text-sm text-emerald-700 dark:text-emerald-200">
{status.message}
</div>
)}
<div className="flex items-center justify-between gap-4 flex-wrap">
<div className="text-sm text-gray-500 dark:text-gray-400">
Konto-ID: <span className="font-mono">{session?.user?.id || '—'}</span>
</div>
<div className="flex items-center gap-3 flex-wrap">
<button
type="button"
onClick={handleDelete}
disabled={deleting || saving}
className="px-5 py-3 rounded-lg border border-rose-300 bg-white text-rose-700 font-semibold hover:bg-rose-50 transition disabled:opacity-60 disabled:cursor-not-allowed dark:border-rose-800 dark:bg-gray-900 dark:text-rose-300 dark:hover:bg-rose-950/40"
>
{deleting ? 'Löscht...' : 'Profil löschen'}
</button>
<button
type="submit"
disabled={saving || deleting}
className="px-5 py-3 rounded-lg bg-sky-600 text-white font-semibold hover:bg-sky-700 transition disabled:opacity-60 disabled:cursor-not-allowed"
>
{saving ? 'Speichert...' : 'Profil speichern'}
</button>
</div>
</div>
</form>
)}
</div>
</div>
</div>
);
}