Enhance Login component with improved validation, loading state, and UI updates

This commit is contained in:
Atharva Ombase
2025-08-03 19:27:07 +05:30
parent dff2de5ddd
commit 0cd4738d09
+122 -90
View File
@@ -1,78 +1,76 @@
import { useState, useEffect } from "react"; import { useState, useEffect } from "react";
import { FiEye, FiEyeOff } from "react-icons/fi"; import { FiEye, FiEyeOff, FiLoader } from "react-icons/fi";
import { Link, useNavigate } from "react-router-dom"; import { Link, useNavigate } from "react-router-dom";
import toast from "react-hot-toast"; // Import React Hot Toast import toast, { Toaster } from "react-hot-toast"; // Import React Hot Toast
import { useTranslation } from "react-i18next"; // for multilinguality import { useTranslation } from "react-i18next"; // for multilinguality
const API_URL = import.meta.env.VITE_API_URL; // Using .env variable const API_URL = import.meta.env.VITE_API_URL; // Using .env variable
const Login = () => { const Login = () => {
const { t } = useTranslation(); // for multilinguality const { t } = useTranslation(); // for multilinguality
const [showPassword, setShowPassword] = useState(false);
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const [loading, setLoading] = useState(false);
const navigate = useNavigate(); // For navigation const navigate = useNavigate(); // For navigation
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const [showPassword, setShowPassword] = useState(false);
const [loading, setLoading] = useState(false);
const [errors, setErrors] = useState({});
// Redirect if already logged in
useEffect(() => { useEffect(() => {
// Check if token is present in localStorage and redirect to Dashboard
if (localStorage.getItem("token")) { if (localStorage.getItem("token")) {
navigate("/dashboard"); // Redirect to Dashboard navigate("/dashboard");
} }
}, [navigate]); }, [navigate]);
const togglePassword = () => { const togglePassword = () => setShowPassword((prev) => !prev);
setShowPassword(!showPassword);
const validate = () => {
const errs = {};
if (!email.trim()) errs.email = t("email_required");
else if (!/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(email))
errs.email = t("invalid_email");
if (!password) errs.password = t("password_required");
return errs;
}; };
const handleSubmit = async (e) => { const handleSubmit = async (e) => {
e.preventDefault(); e.preventDefault();
setLoading(true); const validation = validate();
if (Object.keys(validation).length) {
setErrors(validation);
return;
}
// Show loading toast setLoading(true);
const toastId = toast.loading(t("logging_in_toast")); const toastId = toast.loading(t("logging_in_toast"));
try { try {
const response = await fetch(`${API_URL}/api/login`, { const response = await fetch(`${API_URL}/api/auth/login`, {
method: "POST", method: "POST",
headers: { headers: { "Content-Type": "application/json" },
"Content-Type": "application/json", body: JSON.stringify({ email, password }),
},
body: JSON.stringify({
email,
password,
}),
}); });
const data = await response.json(); const data = await response.json();
// Dismiss the loading toast after the response
toast.dismiss(toastId); toast.dismiss(toastId);
if (response.ok) { if (response.ok) {
// On success, store the token in localStorage
localStorage.setItem("token", data.token); localStorage.setItem("token", data.token);
localStorage.setItem("expiresIn", data.expiresIn); localStorage.setItem("expiresIn", data.expiresIn);
// fetch username asynchronously
fetch(`${API_URL}/api/hdfs/getUsernameByEmail?email=${email}`) fetch(`${API_URL}/api/hdfs/getUsernameByEmail?email=${email}`)
.then((response) => response.text()) .then((res) => res.text())
.then((username) => { .then((username) => localStorage.setItem("username", username))
localStorage.setItem("username", username); .catch((err) => console.error("Error fetching username:", err));
})
.catch((error) => {
console.error("Error fetching username:", error);
});
// Show success toast
toast.success(t("login_successful")); toast.success(t("login_successful"));
// Redirect to Dashboard
navigate("/dashboard"); navigate("/dashboard");
} else { } else {
// Show error toast if login fails
toast.error(data.message || t("login_failed")); toast.error(data.message || t("login_failed"));
} }
} catch (error) { } catch (error) {
// Dismiss the loading toast and show error
toast.dismiss(toastId); toast.dismiss(toastId);
console.error(error);
toast.error(t("an_error_occurred")); toast.error(t("an_error_occurred"));
} finally { } finally {
setLoading(false); setLoading(false);
@@ -80,73 +78,107 @@ const Login = () => {
}; };
return ( return (
<div className="min-h-screen bg-gray-100 flex items-center justify-center p-4"> <div className="min-h-screen bg-gray-50 flex items-center justify-center p-4">
<div className="w-full max-w-md bg-white rounded-4xl shadow-lg p-8"> <Toaster position="top-right" />
<h1 className="text-2xl font-bold mb-6 text-gray-900 text-center"> <div className="w-full max-w-sm bg-white rounded-2xl shadow-lg p-8">
<h1 className="text-2xl font-bold text-gray-800 mb-6 text-center">
{t("login_title")} {t("login_title")}
</h1> </h1>
<form onSubmit={handleSubmit} noValidate className="space-y-5">
<form onSubmit={handleSubmit}> {/* Email Field */}
<div className="mb-4"> <div>
<div className="flex items-center"> <label
<input htmlFor="email"
type="email" className="block text-sm font-medium text-gray-700 mb-1"
id="email"
placeholder={t("email_placeholder")}
className="w-full border border-gray-300 rounded-l-lg px-4 py-4 focus:outline-none focus:border-blue-500"
value={email}
onChange={(e) => setEmail(e.target.value)}
required
/>
</div>
</div>
<div className="mb-1">
<div className="relative">
<input
type={showPassword ? "text" : "password"}
id="password"
placeholder={t("password_placeholder")}
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:border-blue-500 pr-10"
value={password}
onChange={(e) => setPassword(e.target.value)}
required
/>
<button
type="button"
onClick={togglePassword}
className="absolute right-2 top-4 text-2xl text-gray-500 hover:text-gray-700"
>
{showPassword ? <FiEyeOff /> : <FiEye />}
</button>
</div>
</div>
<div className="mb-6 ">
<Link
to="#!"
className="text-sm text-blue-600 hover:underline inline-block"
> >
{t("email_placeholder")}
</label>
<input
type="email"
id="email"
value={email}
onChange={(e) => {
setEmail(e.target.value);
setErrors((prev) => ({ ...prev, email: undefined }));
}}
className={`w-full border ${
errors.email ? "border-red-500" : "border-gray-300"
} rounded-lg px-4 py-2 focus:outline-none focus:ring-2 focus:ring-blue-500`}
placeholder={t("email_placeholder")}
required
/>
{errors.email && (
<p className="text-red-500 text-xs mt-1">{errors.email}</p>
)}
</div>
{/* Password Field */}
<div className="relative">
<label
htmlFor="password"
className="block text-sm font-medium text-gray-700 mb-1"
>
{t("password_placeholder")}
</label>
<input
type={showPassword ? "text" : "password"}
id="password"
value={password}
onChange={(e) => {
setPassword(e.target.value);
setErrors((prev) => ({ ...prev, password: undefined }));
}}
className={`w-full border ${
errors.password ? "border-red-500" : "border-gray-300"
} rounded-lg px-4 py-2 pr-10 focus:outline-none focus:ring-2 focus:ring-blue-500`}
placeholder={t("password_placeholder")}
required
/>
<button
type="button"
onClick={togglePassword}
className="absolute right-3 top-8 text-xl text-gray-500 hover:text-gray-700"
>
{showPassword ? <FiEyeOff /> : <FiEye />}
</button>
{errors.password && (
<p className="text-red-500 text-xs mt-1">{errors.password}</p>
)}
</div>
{/* Forgot & Submit */}
<div className="flex items-center justify-between">
<Link to="#!" className="text-sm text-blue-600 hover:underline">
{t("forgot_password")} {t("forgot_password")}
</Link> </Link>
</div> </div>
<button <button
type="submit" type="submit"
disabled={loading} disabled={loading}
className="w-full py-3 bg-gradient-to-r from-[#1877F2] to-[#0E458C] hover:from-[#0E458C] hover:to-[#1877F2] text-white font-semibold rounded-full shadow-md transition duration-300" className={`w-full flex justify-center items-center py-3 ${
loading
? "bg-gray-400 cursor-not-allowed"
: "bg-gradient-to-r from-blue-600 to-blue-800 hover:from-blue-700 hover:to-blue-900"
} text-white font-semibold rounded-lg shadow-md transition duration-300`}
> >
{loading ? t("logging_in") : t("login")} {loading ? (
<FiLoader className="animate-spin text-lg" />
) : (
t("login")
)}
</button> </button>
</form> </form>
<div className="text-center mt-6">
<p className="text-gray-700"> <p className="text-center mt-6 text-gray-600">
{t("dont_have_account")}{" "} {t("dont_have_account")}{" "}
<Link <Link
to="/signup" to="/signup"
className="text-emerald-500 hover:underline font-medium" className="text-green-600 hover:underline font-medium"
> >
{t("sign_up")} {t("sign_up")}
</Link> </Link>
</p> </p>
</div>
</div> </div>
</div> </div>
); );