Files
Momento/memento-note/components/login-form.tsx
Antigravity 09a63c487d
All checks were successful
Deploy to Production / Build and Deploy (push) Successful in 5s
feat: redesign auth pages — rounded-[48px] cards, serif titles, orbs, icon inputs
- 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
2026-05-16 20:41:28 +00:00

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>
);
}