owlcub-academy/src/app/[locale]/profile/PasswordForm.tsx

122 lines
3.7 KiB
TypeScript

"use client";
import { useState } from "react";
export function PasswordForm({ hasPassword }: { hasPassword: boolean }) {
const [currentPassword, setCurrentPassword] = useState("");
const [newPassword, setNewPassword] = useState("");
const [confirmPassword, setConfirmPassword] = useState("");
const [loading, setLoading] = useState(false);
const [error, setError] = useState("");
const [success, setSuccess] = useState(false);
const handleSubmit = async (e: React.FormEvent) => {
e.preventDefault();
setError("");
setSuccess(false);
if (newPassword !== confirmPassword) {
setError("Les mots de passe ne correspondent pas.");
return;
}
if (newPassword.length < 8) {
setError("Le mot de passe doit contenir au moins 8 caractères.");
return;
}
setLoading(true);
try {
const res = await fetch("/api/profile/password", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
currentPassword: hasPassword ? currentPassword : undefined,
newPassword,
}),
});
const data = await res.json();
if (!res.ok) {
setError(data.error || "Une erreur est survenue.");
} else {
setSuccess(true);
setCurrentPassword("");
setNewPassword("");
setConfirmPassword("");
}
} catch {
setError("Une erreur est survenue.");
} finally {
setLoading(false);
}
};
return (
<form onSubmit={handleSubmit} style={{ display: "flex", flexDirection: "column", gap: 16 }}>
{hasPassword && (
<div>
<label style={{ display: "block", fontSize: 13, fontWeight: 600, color: "#94a3b8", marginBottom: 6 }}>
Mot de passe actuel
</label>
<input
type="password"
value={currentPassword}
onChange={(e) => setCurrentPassword(e.target.value)}
required
autoComplete="current-password"
/>
</div>
)}
<div>
<label style={{ display: "block", fontSize: 13, fontWeight: 600, color: "#94a3b8", marginBottom: 6 }}>
Nouveau mot de passe
</label>
<input
type="password"
value={newPassword}
onChange={(e) => setNewPassword(e.target.value)}
placeholder="8 caractères minimum"
required
autoComplete="new-password"
/>
</div>
<div>
<label style={{ display: "block", fontSize: 13, fontWeight: 600, color: "#94a3b8", marginBottom: 6 }}>
Confirmer le mot de passe
</label>
<input
type="password"
value={confirmPassword}
onChange={(e) => setConfirmPassword(e.target.value)}
placeholder="Répétez le mot de passe"
required
autoComplete="new-password"
/>
</div>
{error && (
<p style={{ fontSize: 13, color: "#f87171", background: "rgba(239,68,68,0.1)", border: "1px solid rgba(239,68,68,0.3)", borderRadius: 6, padding: "10px 12px" }}>
{error}
</p>
)}
{success && (
<p style={{ fontSize: 13, color: "#4ade80", background: "rgba(34,197,94,0.1)", border: "1px solid rgba(34,197,94,0.3)", borderRadius: 6, padding: "10px 12px" }}>
Mot de passe mis à jour avec succès.
</p>
)}
<button
type="submit"
disabled={loading}
className="btn btn-primary"
style={{ justifyContent: "center", opacity: loading ? 0.7 : 1, cursor: loading ? "not-allowed" : "pointer" }}
>
{loading ? "Enregistrement…" : hasPassword ? "Changer le mot de passe" : "Définir le mot de passe"}
</button>
</form>
);
}