217 lines
7.4 KiB
TypeScript
217 lines
7.4 KiB
TypeScript
import React, { useState, useEffect } from "react";
|
||
import { useNavigate, Link } from "react-router-dom";
|
||
import { FiUser, FiLock, FiLogIn } from "react-icons/fi";
|
||
import { useAuthStore } from "@/modules/auth/store/useAuthStore";
|
||
|
||
export const AuthPage: React.FC = () => {
|
||
const navigate = useNavigate();
|
||
const { login, isLoading, error, clearError, token } = useAuthStore();
|
||
const [formData, setFormData] = useState({
|
||
login: "",
|
||
password: "",
|
||
});
|
||
|
||
useEffect(() => {
|
||
if (token) {
|
||
navigate("/");
|
||
}
|
||
}, [token, navigate]);
|
||
|
||
const handleSubmit = async (e: React.FormEvent) => {
|
||
e.preventDefault();
|
||
try {
|
||
await login(formData);
|
||
navigate("/");
|
||
} catch (err) {
|
||
// Error is handled by store
|
||
}
|
||
};
|
||
|
||
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
||
setFormData({
|
||
...formData,
|
||
[e.target.name]: e.target.value,
|
||
});
|
||
if (error) clearError();
|
||
};
|
||
|
||
return (
|
||
<div
|
||
className="min-h-screen flex items-center justify-center p-4"
|
||
style={{ backgroundColor: "var(--bg-primary)" }}
|
||
>
|
||
<div className="w-full max-w-md">
|
||
{/* Card */}
|
||
<div
|
||
className="rounded-2xl shadow-2xl p-8 border"
|
||
style={{
|
||
backgroundColor: "var(--card-bg)",
|
||
borderColor: "var(--border)",
|
||
boxShadow: "0 20px 60px var(--shadow-color)",
|
||
}}
|
||
>
|
||
{/* Header */}
|
||
<div className="text-center mb-8">
|
||
<div
|
||
className="w-16 h-16 mx-auto mb-4 rounded-full flex items-center justify-center"
|
||
style={{ backgroundColor: "var(--bg-secondary)" }}
|
||
>
|
||
<FiUser className="w-8 h-8" style={{ color: "var(--accent)" }} />
|
||
</div>
|
||
<h1
|
||
className="text-3xl font-bold mb-2"
|
||
style={{ color: "var(--text-primary)" }}
|
||
>
|
||
С возвращением!
|
||
</h1>
|
||
<p style={{ color: "var(--text-secondary)" }}>
|
||
Войдите в свой аккаунт
|
||
</p>
|
||
</div>
|
||
|
||
{/* Error Message */}
|
||
{error && (
|
||
<div
|
||
className="mb-6 p-4 rounded-lg border text-sm"
|
||
style={{
|
||
backgroundColor: "var(--error-bg)",
|
||
borderColor: "var(--error-border)",
|
||
color: "var(--error-text)",
|
||
}}
|
||
>
|
||
{error}
|
||
</div>
|
||
)}
|
||
|
||
{/* Form */}
|
||
<form onSubmit={handleSubmit} className="space-y-5">
|
||
<div>
|
||
<label
|
||
className="block text-sm font-medium mb-2"
|
||
style={{ color: "var(--text-secondary)" }}
|
||
>
|
||
Логин
|
||
</label>
|
||
<div className="relative">
|
||
<FiUser
|
||
className="absolute left-3 top-1/2 transform -translate-y-1/2"
|
||
style={{ color: "var(--text-muted)" }}
|
||
/>
|
||
<input
|
||
type="text"
|
||
name="login"
|
||
value={formData.login}
|
||
onChange={handleChange}
|
||
required
|
||
className="w-full pl-10 pr-3 py-2.5 rounded-lg border focus:outline-none focus:ring-2 transition-all"
|
||
style={{
|
||
backgroundColor: "var(--input-bg)",
|
||
borderColor: "var(--border)",
|
||
color: "var(--text-primary)",
|
||
}}
|
||
onFocus={(e) => {
|
||
e.currentTarget.style.borderColor = "var(--border-focus)";
|
||
e.currentTarget.style.boxShadow = `0 0 0 3px ${e.currentTarget.style.borderColor}20`;
|
||
}}
|
||
onBlur={(e) => {
|
||
e.currentTarget.style.borderColor = "var(--border)";
|
||
e.currentTarget.style.boxShadow = "none";
|
||
}}
|
||
placeholder="Введите ваш логин"
|
||
/>
|
||
</div>
|
||
</div>
|
||
|
||
<div>
|
||
<label
|
||
className="block text-sm font-medium mb-2"
|
||
style={{ color: "var(--text-secondary)" }}
|
||
>
|
||
Пароль
|
||
</label>
|
||
<div className="relative">
|
||
<FiLock
|
||
className="absolute left-3 top-1/2 transform -translate-y-1/2"
|
||
style={{ color: "var(--text-muted)" }}
|
||
/>
|
||
<input
|
||
type="password"
|
||
name="password"
|
||
value={formData.password}
|
||
onChange={handleChange}
|
||
required
|
||
className="w-full pl-10 pr-3 py-2.5 rounded-lg border focus:outline-none focus:ring-2 transition-all"
|
||
style={{
|
||
backgroundColor: "var(--input-bg)",
|
||
borderColor: "var(--border)",
|
||
color: "var(--text-primary)",
|
||
}}
|
||
onFocus={(e) => {
|
||
e.currentTarget.style.borderColor = "var(--border-focus)";
|
||
e.currentTarget.style.boxShadow = `0 0 0 3px ${e.currentTarget.style.borderColor}20`;
|
||
}}
|
||
onBlur={(e) => {
|
||
e.currentTarget.style.borderColor = "var(--border)";
|
||
e.currentTarget.style.boxShadow = "none";
|
||
}}
|
||
placeholder="Введите ваш пароль"
|
||
/>
|
||
</div>
|
||
</div>
|
||
|
||
<button
|
||
type="submit"
|
||
disabled={isLoading}
|
||
className="w-full flex items-center justify-center gap-2 px-4 py-2.5 rounded-lg transition-colors disabled:opacity-50 disabled:cursor-not-allowed font-medium"
|
||
style={{
|
||
backgroundColor: "var(--button-primary)",
|
||
color: "var(--button-primary-text)",
|
||
}}
|
||
onMouseEnter={(e) => {
|
||
if (!isLoading) {
|
||
e.currentTarget.style.backgroundColor = "var(--button-primary-hover)";
|
||
}
|
||
}}
|
||
onMouseLeave={(e) => {
|
||
e.currentTarget.style.backgroundColor = "var(--button-primary)";
|
||
}}
|
||
>
|
||
{isLoading ? (
|
||
<>
|
||
<div className="w-5 h-5 border-2 border-current border-t-transparent rounded-full animate-spin" />
|
||
Вход...
|
||
</>
|
||
) : (
|
||
<>
|
||
<FiLogIn />
|
||
Войти
|
||
</>
|
||
)}
|
||
</button>
|
||
</form>
|
||
|
||
{/* Footer */}
|
||
<div className="mt-6 text-center">
|
||
<p className="text-sm" style={{ color: "var(--text-secondary)" }}>
|
||
Нет аккаунта?{" "}
|
||
<Link
|
||
to="/register"
|
||
className="font-medium hover:underline transition-colors"
|
||
style={{ color: "var(--link)" }}
|
||
onMouseEnter={(e) => {
|
||
e.currentTarget.style.color = "var(--link-hover)";
|
||
}}
|
||
onMouseLeave={(e) => {
|
||
e.currentTarget.style.color = "var(--link)";
|
||
}}
|
||
>
|
||
Зарегистрироваться
|
||
</Link>
|
||
</p>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
);
|
||
};
|