Performed translation for Authentication in src/pages/
- Login.jsx - SignUp.jsx Also, added key:value pairs (English + French for now) for all the text in src/locales.
This commit is contained in:
@@ -28,7 +28,37 @@
|
|||||||
|
|
||||||
"not_found_title": "Page Not Found",
|
"not_found_title": "Page Not Found",
|
||||||
"not_found_description": "Sorry, we couldn't find the page you were looking for. It may have been moved or deleted.",
|
"not_found_description": "Sorry, we couldn't find the page you were looking for. It may have been moved or deleted.",
|
||||||
"go_home": "Go Home"
|
"go_home": "Go Home",
|
||||||
|
|
||||||
|
"login_title": "Log in",
|
||||||
|
"email_placeholder": "Enter your email",
|
||||||
|
"password_placeholder": "Enter your password",
|
||||||
|
"forgot_password": "Forgot password?",
|
||||||
|
"logging_in": "Logging In...",
|
||||||
|
"login": "Login",
|
||||||
|
"dont_have_account": "Don’t have an account?",
|
||||||
|
"sign_up": "Sign up",
|
||||||
|
"login_successful": "Login successful!",
|
||||||
|
"login_failed": "Login failed.",
|
||||||
|
"an_error_occurred": "An error occurred. Please try again.",
|
||||||
|
"logging_in_toast": "Logging in...",
|
||||||
|
|
||||||
|
"signup_title": "Sign Up",
|
||||||
|
"first_name": "First Name",
|
||||||
|
"last_name": "Last Name",
|
||||||
|
"email_placeholder": "Enter your email",
|
||||||
|
"password_placeholder": "Enter your password",
|
||||||
|
"confirm_password_placeholder": "Confirm your password",
|
||||||
|
"signing_up": "Signing Up...",
|
||||||
|
"sign_up": "Sign Up",
|
||||||
|
"already_have_account": "Already have an account?",
|
||||||
|
"login": "Login",
|
||||||
|
"passwords_do_not_match": "Passwords do not match.",
|
||||||
|
"registering": "Registering...",
|
||||||
|
"signup_failed": "Signup failed.",
|
||||||
|
"folder_creation_failed": "Failed to create user folder.",
|
||||||
|
"signup_success": "Successfully registered and folder created!",
|
||||||
|
"an_error_occurred": "An error occurred. Please try again."
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,7 +28,36 @@
|
|||||||
|
|
||||||
"not_found_title": "Page non trouvée",
|
"not_found_title": "Page non trouvée",
|
||||||
"not_found_description": "Désolé, nous n'avons pas pu trouver la page que vous cherchiez. Elle a peut-être été déplacée ou supprimée.",
|
"not_found_description": "Désolé, nous n'avons pas pu trouver la page que vous cherchiez. Elle a peut-être été déplacée ou supprimée.",
|
||||||
"go_home": "Accueil"
|
"go_home": "Accueil",
|
||||||
|
|
||||||
|
"login_title": "Connexion",
|
||||||
|
"email_placeholder": "Entrez votre e-mail",
|
||||||
|
"password_placeholder": "Entrez votre mot de passe",
|
||||||
|
"forgot_password": "Mot de passe oublié ?",
|
||||||
|
"logging_in": "Connexion...",
|
||||||
|
"login": "Connexion",
|
||||||
|
"dont_have_account": "Vous n'avez pas de compte ?",
|
||||||
|
"sign_up": "S'inscrire",
|
||||||
|
"login_successful": "Connexion réussie !",
|
||||||
|
"login_failed": "Échec de la connexion.",
|
||||||
|
"an_error_occurred": "Une erreur s'est produite. Veuillez réessayer.",
|
||||||
|
"logging_in_toast": "Connexion en cours...",
|
||||||
|
|
||||||
|
"sign_up": "S'inscrire",
|
||||||
|
"first_name": "Prénom",
|
||||||
|
"last_name": "Nom de famille",
|
||||||
|
"email_placeholder": "Entrez votre e-mail",
|
||||||
|
"password_placeholder": "Entrez votre mot de passe",
|
||||||
|
"confirm_password_placeholder": "Confirmez votre mot de passe",
|
||||||
|
"already_have_account": "Vous avez déjà un compte ?",
|
||||||
|
"login": "Connexion",
|
||||||
|
"signing_up": "Inscription...",
|
||||||
|
"passwords_do_not_match": "Les mots de passe ne correspondent pas.",
|
||||||
|
"registering": "Enregistrement...",
|
||||||
|
"signup_failed": "Échec de l'inscription.",
|
||||||
|
"failed_create_folder": "Échec de la création du dossier utilisateur.",
|
||||||
|
"signup_success": "Inscription réussie et dossier créé !",
|
||||||
|
"an_error_occurred": "Une erreur s'est produite. Veuillez réessayer."
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,10 +2,12 @@ import { useState, useEffect } from "react";
|
|||||||
import { FiEye, FiEyeOff } from "react-icons/fi";
|
import { FiEye, FiEyeOff } 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 from "react-hot-toast"; // Import React Hot Toast
|
||||||
|
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 [showPassword, setShowPassword] = useState(false);
|
const [showPassword, setShowPassword] = useState(false);
|
||||||
const [email, setEmail] = useState("");
|
const [email, setEmail] = useState("");
|
||||||
const [password, setPassword] = useState("");
|
const [password, setPassword] = useState("");
|
||||||
@@ -28,7 +30,7 @@ const Login = () => {
|
|||||||
setLoading(true);
|
setLoading(true);
|
||||||
|
|
||||||
// Show loading toast
|
// Show loading toast
|
||||||
const toastId = toast.loading("Logging in...");
|
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/login`, {
|
||||||
@@ -61,18 +63,17 @@ const Login = () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Show success toast
|
// Show success toast
|
||||||
toast.success("Login successful!");
|
toast.success(t("login_successful"));
|
||||||
|
|
||||||
// Redirect to Dashboard
|
// Redirect to Dashboard
|
||||||
navigate("/dashboard");
|
navigate("/dashboard");
|
||||||
} else {
|
} else {
|
||||||
// Show error toast if login fails
|
// Show error toast if login fails
|
||||||
toast.error(data.message || "Login failed.");
|
toast.error(data.message || t("login_failed"));
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// Dismiss the loading toast and show error
|
// Dismiss the loading toast and show error
|
||||||
toast.dismiss(toastId);
|
toast.dismiss(toastId);
|
||||||
toast.error("An error occurred. Please try again.", error);
|
toast.error(t("an_error_occurred"));
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
@@ -82,7 +83,7 @@ const Login = () => {
|
|||||||
<div className="min-h-screen bg-gray-100 flex items-center justify-center p-4">
|
<div className="min-h-screen bg-gray-100 flex items-center justify-center p-4">
|
||||||
<div className="w-full max-w-md bg-white rounded-4xl shadow-lg p-8">
|
<div className="w-full max-w-md bg-white rounded-4xl shadow-lg p-8">
|
||||||
<h1 className="text-2xl font-bold mb-6 text-gray-900 text-center">
|
<h1 className="text-2xl font-bold mb-6 text-gray-900 text-center">
|
||||||
Log in
|
{t("login_title")}
|
||||||
</h1>
|
</h1>
|
||||||
|
|
||||||
<form onSubmit={handleSubmit}>
|
<form onSubmit={handleSubmit}>
|
||||||
@@ -91,7 +92,7 @@ const Login = () => {
|
|||||||
<input
|
<input
|
||||||
type="email"
|
type="email"
|
||||||
id="email"
|
id="email"
|
||||||
placeholder="Enter your 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"
|
className="w-full border border-gray-300 rounded-l-lg px-4 py-4 focus:outline-none focus:border-blue-500"
|
||||||
value={email}
|
value={email}
|
||||||
onChange={(e) => setEmail(e.target.value)}
|
onChange={(e) => setEmail(e.target.value)}
|
||||||
@@ -104,7 +105,7 @@ const Login = () => {
|
|||||||
<input
|
<input
|
||||||
type={showPassword ? "text" : "password"}
|
type={showPassword ? "text" : "password"}
|
||||||
id="password"
|
id="password"
|
||||||
placeholder="Enter your 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"
|
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:border-blue-500 pr-10"
|
||||||
value={password}
|
value={password}
|
||||||
onChange={(e) => setPassword(e.target.value)}
|
onChange={(e) => setPassword(e.target.value)}
|
||||||
@@ -124,7 +125,7 @@ const Login = () => {
|
|||||||
to="#!"
|
to="#!"
|
||||||
className="text-sm text-blue-600 hover:underline inline-block"
|
className="text-sm text-blue-600 hover:underline inline-block"
|
||||||
>
|
>
|
||||||
Forgot password?
|
{t("forgot_password")}
|
||||||
</Link>
|
</Link>
|
||||||
</div>
|
</div>
|
||||||
<button
|
<button
|
||||||
@@ -132,17 +133,17 @@ const Login = () => {
|
|||||||
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 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"
|
||||||
>
|
>
|
||||||
{loading ? "Logging In..." : "Login"}
|
{loading ? t("logging_in") : t("login")}
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
<div className="text-center mt-6">
|
<div className="text-center mt-6">
|
||||||
<p className="text-gray-700">
|
<p className="text-gray-700">
|
||||||
Don’t have an account?{" "}
|
{t("dont_have_account")}{" "}
|
||||||
<Link
|
<Link
|
||||||
to="/signup"
|
to="/signup"
|
||||||
className="text-emerald-500 hover:underline font-medium"
|
className="text-emerald-500 hover:underline font-medium"
|
||||||
>
|
>
|
||||||
Sign up
|
{t("sign_up")}
|
||||||
</Link>
|
</Link>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -2,10 +2,12 @@ import { useState } from "react";
|
|||||||
import { FiEye, FiEyeOff } from "react-icons/fi";
|
import { FiEye, FiEyeOff } from "react-icons/fi";
|
||||||
import { Link, useNavigate } from "react-router-dom";
|
import { Link, useNavigate } from "react-router-dom";
|
||||||
import toast, { Toaster } from "react-hot-toast";
|
import toast, { Toaster } from "react-hot-toast";
|
||||||
|
import { useTranslation } from "react-i18next"; // for multilinguality
|
||||||
|
|
||||||
const API_URL = import.meta.env.VITE_API_URL;
|
const API_URL = import.meta.env.VITE_API_URL;
|
||||||
|
|
||||||
const SignUp = () => {
|
const SignUp = () => {
|
||||||
|
const { t } = useTranslation(); // for multilinguality
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
const [formData, setFormData] = useState({
|
const [formData, setFormData] = useState({
|
||||||
@@ -30,12 +32,12 @@ const SignUp = () => {
|
|||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
if (formData.password !== formData.confirmPassword) {
|
if (formData.password !== formData.confirmPassword) {
|
||||||
toast.error("Passwords do not match.");
|
toast.error(t("passwords_do_not_match"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
const toastId = toast.loading("Registering...");
|
const toastId = toast.loading(t("registering"));
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// 1️⃣ Sign up the user
|
// 1️⃣ Sign up the user
|
||||||
@@ -52,7 +54,7 @@ const SignUp = () => {
|
|||||||
const signupData = await signupRes.json();
|
const signupData = await signupRes.json();
|
||||||
|
|
||||||
if (!signupRes.ok) {
|
if (!signupRes.ok) {
|
||||||
toast.error(signupData.message || "Signup failed.", { id: toastId });
|
toast.error(signupData.message || t("signup_failed"), { id: toastId });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -66,11 +68,9 @@ const SignUp = () => {
|
|||||||
|
|
||||||
if (!folderRes.ok) {
|
if (!folderRes.ok) {
|
||||||
// you might choose to roll back user creation or just notify
|
// you might choose to roll back user creation or just notify
|
||||||
toast.error("Failed to create user folder.", { id: toastId });
|
toast.error(t("failed_create_folder"), { id: toastId });
|
||||||
} else {
|
} else {
|
||||||
toast.success("Successfully registered and folder created!", {
|
toast.success(t("signup_success"), { id: toastId });
|
||||||
id: toastId,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3️⃣ Redirect to login after a short delay
|
// 3️⃣ Redirect to login after a short delay
|
||||||
@@ -79,7 +79,7 @@ const SignUp = () => {
|
|||||||
}, 1500);
|
}, 1500);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
toast.error("An error occurred. Please try again.", { id: toastId });
|
toast.error(t("an_error_occurred"), { id: toastId });
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
@@ -89,12 +89,12 @@ const SignUp = () => {
|
|||||||
<div className="min-h-screen bg-gray-100 flex items-center justify-center p-6">
|
<div className="min-h-screen bg-gray-100 flex items-center justify-center p-6">
|
||||||
<Toaster position="top-right" />
|
<Toaster position="top-right" />
|
||||||
<div className="w-full max-w-md bg-white rounded-2xl shadow-lg p-8">
|
<div className="w-full max-w-md bg-white rounded-2xl shadow-lg p-8">
|
||||||
<h1 className="text-2xl font-bold text-gray-900 mb-6">Sign Up</h1>
|
<h1 className="text-2xl font-bold text-gray-900 mb-6">{t("sign_up")}</h1>
|
||||||
<form className="space-y-4" onSubmit={handleSubmit}>
|
<form className="space-y-4" onSubmit={handleSubmit}>
|
||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
name="firstname"
|
name="firstname"
|
||||||
placeholder="First Name"
|
placeholder={t("first_name")}
|
||||||
value={formData.firstname}
|
value={formData.firstname}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
|
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
|
||||||
@@ -103,7 +103,7 @@ const SignUp = () => {
|
|||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
name="lastname"
|
name="lastname"
|
||||||
placeholder="Last Name"
|
placeholder={t("last_name")}
|
||||||
value={formData.lastname}
|
value={formData.lastname}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
|
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
|
||||||
@@ -112,7 +112,7 @@ const SignUp = () => {
|
|||||||
<input
|
<input
|
||||||
type="email"
|
type="email"
|
||||||
name="email"
|
name="email"
|
||||||
placeholder="Enter your email"
|
placeholder={t("email_placeholder")}
|
||||||
value={formData.email}
|
value={formData.email}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
|
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500"
|
||||||
@@ -124,7 +124,7 @@ const SignUp = () => {
|
|||||||
<input
|
<input
|
||||||
type={showPassword ? "text" : "password"}
|
type={showPassword ? "text" : "password"}
|
||||||
name="password"
|
name="password"
|
||||||
placeholder="Enter your password"
|
placeholder={t("password_placeholder")}
|
||||||
value={formData.password}
|
value={formData.password}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500 pr-10"
|
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500 pr-10"
|
||||||
@@ -144,7 +144,7 @@ const SignUp = () => {
|
|||||||
<input
|
<input
|
||||||
type={showConfirmPassword ? "text" : "password"}
|
type={showConfirmPassword ? "text" : "password"}
|
||||||
name="confirmPassword"
|
name="confirmPassword"
|
||||||
placeholder="Confirm your password"
|
placeholder={t("confirm_password_placeholder")}
|
||||||
value={formData.confirmPassword}
|
value={formData.confirmPassword}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500 pr-10"
|
className="w-full border border-gray-300 rounded-lg px-4 py-4 focus:outline-none focus:ring-2 focus:ring-blue-500 pr-10"
|
||||||
@@ -169,18 +169,18 @@ const SignUp = () => {
|
|||||||
: "bg-gradient-to-r from-[#10B981] to-[#07533A] hover:from-[#0E458C] hover:to-[#1877F2]"
|
: "bg-gradient-to-r from-[#10B981] to-[#07533A] hover:from-[#0E458C] hover:to-[#1877F2]"
|
||||||
} text-white font-semibold rounded-lg shadow-md transition duration-300`}
|
} text-white font-semibold rounded-lg shadow-md transition duration-300`}
|
||||||
>
|
>
|
||||||
{loading ? "Signing Up..." : "Sign Up"}
|
{loading ? t("signing_up") : t("sign_up")}
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
{/* Redirect to Login */}
|
{/* Redirect to Login */}
|
||||||
<p className="text-center mt-4 text-gray-700">
|
<p className="text-center mt-4 text-gray-700">
|
||||||
Already have an account?{" "}
|
{t("already_have_account")}{" "}
|
||||||
<Link
|
<Link
|
||||||
to="/login"
|
to="/login"
|
||||||
className="text-blue-500 hover:underline font-medium"
|
className="text-blue-500 hover:underline font-medium"
|
||||||
>
|
>
|
||||||
Login
|
{t("login")}
|
||||||
</Link>
|
</Link>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user