Press n or j to go to the next uncovered block, b, p or k for the previous block.
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 | 22x 22x 22x 22x 22x 5x 5x 5x 5x 5x 3x 2x 5x 22x 3x 19x 5x | import { useState } from 'react'
import { Link } from 'react-router-dom'
import { motion } from 'framer-motion'
import { Mail, ArrowLeft, Zap, CheckCircle } from 'lucide-react'
import api from '../api/client'
export default function ForgotPassword() {
const [email, setEmail] = useState('')
const [loading, setLoading] = useState(false)
const [error, setError] = useState('')
const [sent, setSent] = useState(false)
const handleSubmit = async (e) => {
e.preventDefault()
setError('')
setLoading(true)
try {
await api.post('/auth/forgot-password', { email })
setSent(true)
} catch (err) {
setError(err.response?.data?.error || 'Ошибка отправки')
} finally {
setLoading(false)
}
}
if (sent) {
return (
<div className="min-h-screen flex items-center justify-center p-4 gradient-mesh bg-surface-50">
<motion.div
initial={{ opacity: 0, scale: 0.9 }}
animate={{ opacity: 1, scale: 1 }}
className="w-full max-w-md"
>
<div className="card p-10 text-center">
<motion.div
initial={{ scale: 0 }}
animate={{ scale: 1 }}
transition={{ type: 'spring', stiffness: 200 }}
className="w-20 h-20 rounded-3xl bg-green-100 flex items-center justify-center mx-auto mb-6"
>
<CheckCircle className="w-10 h-10 text-green-600" />
</motion.div>
<h1 className="text-2xl font-display font-bold text-gray-900 mb-2">
Письмо отправлено! 📬
</h1>
<p className="text-gray-500 mb-6">
Если аккаунт с email <strong>{email}</strong> существует, мы отправили ссылку для сброса пароля.
</p>
<Link to="/login" className="btn btn-primary">
Вернуться ко входу
</Link>
</div>
</motion.div>
</div>
)
}
return (
<div className="min-h-screen flex items-center justify-center p-4 gradient-mesh bg-surface-50">
<motion.div
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
className="w-full max-w-md"
>
<div className="text-center mb-8">
<motion.div
initial={{ scale: 0 }}
animate={{ scale: 1 }}
transition={{ type: 'spring', delay: 0.1 }}
className="inline-flex items-center justify-center w-20 h-20 rounded-3xl bg-gradient-to-br from-primary-500 to-primary-700 mb-6 shadow-xl shadow-primary-500/30"
>
<Mail className="w-10 h-10 text-white" />
</motion.div>
<h1 className="text-3xl font-display font-bold text-gray-900">
Забыли пароль?
</h1>
<p className="text-gray-500 mt-2">
Введи email и мы отправим ссылку для сброса
</p>
</div>
<div className="card p-8">
<form onSubmit={handleSubmit} className="space-y-5">
{error && (
<motion.div
initial={{ opacity: 0, height: 0 }}
animate={{ opacity: 1, height: 'auto' }}
className="p-4 rounded-2xl bg-red-50 text-red-600 text-sm font-medium"
>
{error}
</motion.div>
)}
<div>
<label className="block text-sm font-semibold text-gray-700 mb-2">
Email
</label>
<input
type="email"
value={email}
onChange={(e) => setEmail(e.target.value)}
className="input"
placeholder="your@email.com"
required
autoFocus
/>
</div>
<button
type="submit"
disabled={loading}
className="btn btn-primary w-full text-lg"
>
{loading ? 'Отправляем...' : 'Отправить ссылку'}
</button>
</form>
<div className="mt-6 text-center">
<Link
to="/login"
className="inline-flex items-center gap-2 text-primary-600 hover:text-primary-700 font-medium text-sm"
>
<ArrowLeft size={16} />
Вернуться ко входу
</Link>
</div>
</div>
<div className="flex items-center justify-center gap-2 mt-6 text-gray-400">
<Zap size={16} />
<span className="text-sm font-medium">Pulse</span>
</div>
</motion.div>
</div>
)
}
|