All checks were successful
Deploy to Production / Build and Deploy (push) Successful in 5s
- Auth layout: warm background with brand-accent/ochre orbs, header with Globe + Momento branding - Login form: serif heading, icon-inputs with focus transitions, uppercase labels, submit with arrow - Register form: matching card style with User/Mail/Lock icons, confirm password field - Forgot password: matching card with email icon, success state with mail icon - Reset password: matching card with Lock icons, invalid link state with AlertCircle - All text via i18n (new keys: welcomeBack, createYourSpace, forgot, etc.) - Dark mode support via CSS variables - Removed shadcn Card/Button/Input dependencies from auth forms
125 lines
4.7 KiB
TypeScript
125 lines
4.7 KiB
TypeScript
'use client';
|
|
|
|
import { useActionState } from 'react';
|
|
import { useFormStatus } from 'react-dom';
|
|
import { authenticate } from '@/app/actions/auth';
|
|
import Link from 'next/link';
|
|
import { Mail, Lock, ArrowRight, Sparkles } from 'lucide-react';
|
|
import { useLanguage } from '@/lib/i18n';
|
|
|
|
function LoginButton() {
|
|
const { pending } = useFormStatus();
|
|
const { t } = useLanguage();
|
|
return (
|
|
<button
|
|
type="submit"
|
|
disabled={pending}
|
|
className="w-full bg-[var(--foreground)] text-[var(--background)] py-4 rounded-2xl font-bold uppercase tracking-[0.2em] text-[10px] flex items-center justify-center gap-3 transition-all hover:shadow-xl hover:shadow-black/10 active:scale-[0.98] disabled:opacity-50 mt-4"
|
|
>
|
|
{pending ? (
|
|
<Sparkles size={16} className="animate-spin" />
|
|
) : (
|
|
<>
|
|
{t('auth.signIn')}
|
|
<ArrowRight size={14} />
|
|
</>
|
|
)}
|
|
</button>
|
|
);
|
|
}
|
|
|
|
export function LoginForm({ allowRegister = true }: { allowRegister?: boolean }) {
|
|
const [errorMessage, dispatch] = useActionState(authenticate, undefined);
|
|
const { t } = useLanguage();
|
|
|
|
return (
|
|
<div className="bg-white dark:bg-[var(--background)]/50 border border-[var(--border)] p-8 md:p-10 rounded-[48px] shadow-2xl">
|
|
<div className="space-y-8">
|
|
<div className="text-center space-y-2">
|
|
<h1 className="text-2xl md:text-3xl font-serif font-bold">
|
|
{t('auth.welcomeBack')}
|
|
</h1>
|
|
<p className="text-[var(--muted-foreground)] text-sm font-light">
|
|
{t('auth.welcomeBackSubtitle')}
|
|
</p>
|
|
</div>
|
|
|
|
<form action={dispatch} className="space-y-4">
|
|
<div className="space-y-1.5">
|
|
<label className="text-[10px] uppercase tracking-widest font-bold text-[var(--muted-foreground)] px-4">
|
|
{t('auth.email')}
|
|
</label>
|
|
<div className="relative group">
|
|
<div className="absolute left-4 top-1/2 -translate-y-1/2 text-[var(--muted-foreground)] group-focus-within:text-[var(--color-brand-accent)] transition-colors">
|
|
<Mail size={16} />
|
|
</div>
|
|
<input
|
|
className="w-full bg-slate-50 dark:bg-white/5 border border-[var(--border)] rounded-2xl py-4 pl-12 pr-4 text-sm outline-none focus:border-[var(--color-brand-accent)] focus:ring-4 ring-[var(--color-brand-accent)]/5 transition-all"
|
|
id="email"
|
|
type="email"
|
|
name="email"
|
|
placeholder={t('auth.emailPlaceholder')}
|
|
required
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="space-y-1.5">
|
|
<div className="flex justify-between items-center px-4">
|
|
<label className="text-[10px] uppercase tracking-widest font-bold text-[var(--muted-foreground)]">
|
|
{t('auth.password')}
|
|
</label>
|
|
<Link
|
|
href="/forgot-password"
|
|
className="text-[10px] text-[var(--color-brand-accent)] font-bold uppercase tracking-widest hover:underline"
|
|
>
|
|
{t('auth.forgot')}
|
|
</Link>
|
|
</div>
|
|
<div className="relative group">
|
|
<div className="absolute left-4 top-1/2 -translate-y-1/2 text-[var(--muted-foreground)] group-focus-within:text-[var(--color-brand-accent)] transition-colors">
|
|
<Lock size={16} />
|
|
</div>
|
|
<input
|
|
className="w-full bg-slate-50 dark:bg-white/5 border border-[var(--border)] rounded-2xl py-4 pl-12 pr-4 text-sm outline-none focus:border-[var(--color-brand-accent)] focus:ring-4 ring-[var(--color-brand-accent)]/5 transition-all"
|
|
id="password"
|
|
type="password"
|
|
name="password"
|
|
placeholder="••••••••"
|
|
required
|
|
minLength={6}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<LoginButton />
|
|
|
|
<div
|
|
className="flex h-8 items-end space-x-1"
|
|
aria-live="polite"
|
|
aria-atomic="true"
|
|
>
|
|
{errorMessage && (
|
|
<p className="text-sm text-red-500">{errorMessage}</p>
|
|
)}
|
|
</div>
|
|
</form>
|
|
|
|
{allowRegister && (
|
|
<div className="text-center pt-2">
|
|
<p className="text-xs text-[var(--muted-foreground)]">
|
|
{t('auth.noAccount')}{' '}
|
|
<Link
|
|
href="/register"
|
|
className="text-[var(--color-brand-accent)] font-bold hover:underline"
|
|
>
|
|
{t('auth.signUp')}
|
|
</Link>
|
|
</p>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|