mirror of
https://github.com/samanhappy/mcphub.git
synced 2026-01-11 00:58:20 -05:00
Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: samanhappy <2755122+samanhappy@users.noreply.github.com>
190 lines
6.9 KiB
TypeScript
190 lines
6.9 KiB
TypeScript
import React, { useState } from 'react';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { ChangePasswordCredentials } from '../types';
|
|
import { changePassword } from '../services/authService';
|
|
import { validatePasswordStrength } from '../utils/passwordValidation';
|
|
|
|
interface ChangePasswordFormProps {
|
|
onSuccess?: () => void;
|
|
onCancel?: () => void;
|
|
}
|
|
|
|
const ChangePasswordForm: React.FC<ChangePasswordFormProps> = ({ onSuccess, onCancel }) => {
|
|
const { t } = useTranslation();
|
|
const [formData, setFormData] = useState<ChangePasswordCredentials>({
|
|
currentPassword: '',
|
|
newPassword: '',
|
|
});
|
|
const [confirmPassword, setConfirmPassword] = useState('');
|
|
const [isLoading, setIsLoading] = useState(false);
|
|
const [error, setError] = useState<string | null>(null);
|
|
const [success, setSuccess] = useState(false);
|
|
const [passwordErrors, setPasswordErrors] = useState<string[]>([]);
|
|
|
|
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
const { name, value } = e.target;
|
|
if (name === 'confirmPassword') {
|
|
setConfirmPassword(value);
|
|
} else {
|
|
setFormData(prev => ({ ...prev, [name]: value }));
|
|
|
|
// Validate password strength on change for new password
|
|
if (name === 'newPassword') {
|
|
const validation = validatePasswordStrength(value);
|
|
setPasswordErrors(validation.errors);
|
|
}
|
|
}
|
|
};
|
|
|
|
const handleSubmit = async (e: React.FormEvent) => {
|
|
e.preventDefault();
|
|
setError(null);
|
|
|
|
// Validate password strength
|
|
const validation = validatePasswordStrength(formData.newPassword);
|
|
if (!validation.isValid) {
|
|
setError(t('auth.passwordStrengthError'));
|
|
setPasswordErrors(validation.errors);
|
|
return;
|
|
}
|
|
|
|
// Validate passwords match
|
|
if (formData.newPassword !== confirmPassword) {
|
|
setError(t('auth.passwordsNotMatch'));
|
|
return;
|
|
}
|
|
|
|
setIsLoading(true);
|
|
try {
|
|
const response = await changePassword(formData);
|
|
|
|
if (response.success) {
|
|
setSuccess(true);
|
|
if (onSuccess) {
|
|
onSuccess();
|
|
}
|
|
} else {
|
|
setError(response.message || t('auth.changePasswordError'));
|
|
}
|
|
} catch (err) {
|
|
setError(t('auth.changePasswordError'));
|
|
} finally {
|
|
setIsLoading(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className="p-6 bg-white rounded-lg shadow-md">
|
|
<h2 className="text-xl font-bold mb-4">{t('auth.changePassword')}</h2>
|
|
|
|
{success ? (
|
|
<div className="bg-green-100 border border-green-400 text-green-700 px-4 py-3 rounded mb-4">
|
|
{t('auth.changePasswordSuccess')}
|
|
</div>
|
|
) : (
|
|
<form onSubmit={handleSubmit}>
|
|
{error && (
|
|
<div className="bg-red-100 border border-red-400 text-red-700 px-4 py-3 rounded mb-4">
|
|
{error}
|
|
</div>
|
|
)}
|
|
|
|
<div className="mb-4">
|
|
<label className="block text-gray-700 text-sm font-bold mb-2" htmlFor="currentPassword">
|
|
{t('auth.currentPassword')}
|
|
</label>
|
|
<input
|
|
type="password"
|
|
id="currentPassword"
|
|
name="currentPassword"
|
|
className="w-full p-2 border rounded focus:outline-none focus:ring-2 focus:ring-blue-500 form-input"
|
|
value={formData.currentPassword}
|
|
onChange={handleChange}
|
|
required
|
|
/>
|
|
</div>
|
|
|
|
<div className="mb-4">
|
|
<label className="block text-gray-700 text-sm font-bold mb-2" htmlFor="newPassword">
|
|
{t('auth.newPassword')}
|
|
</label>
|
|
<input
|
|
type="password"
|
|
id="newPassword"
|
|
name="newPassword"
|
|
className="w-full p-2 border rounded focus:outline-none focus:ring-2 focus:ring-blue-500 form-input"
|
|
value={formData.newPassword}
|
|
onChange={handleChange}
|
|
required
|
|
minLength={8}
|
|
/>
|
|
{/* Password strength hints */}
|
|
{formData.newPassword && passwordErrors.length > 0 && (
|
|
<div className="mt-2 text-sm text-gray-600">
|
|
<p className="font-semibold mb-1">{t('auth.passwordStrengthHint')}</p>
|
|
<ul className="list-disc list-inside space-y-1">
|
|
{passwordErrors.map((errorKey) => (
|
|
<li key={errorKey} className="text-red-600">
|
|
{t(`auth.${errorKey}`)}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</div>
|
|
)}
|
|
{formData.newPassword && passwordErrors.length === 0 && (
|
|
<p className="mt-2 text-sm text-green-600">✓ {t('auth.passwordStrengthHint')}</p>
|
|
)}
|
|
</div>
|
|
|
|
<div className="mb-6">
|
|
<label className="block text-gray-700 text-sm font-bold mb-2" htmlFor="confirmPassword">
|
|
{t('auth.confirmPassword')}
|
|
</label>
|
|
<input
|
|
type="password"
|
|
id="confirmPassword"
|
|
name="confirmPassword"
|
|
className="w-full p-2 border rounded focus:outline-none focus:ring-2 focus:ring-blue-500 form-input"
|
|
value={confirmPassword}
|
|
onChange={handleChange}
|
|
required
|
|
minLength={8}
|
|
/>
|
|
</div>
|
|
|
|
<div className="flex justify-end space-x-2">
|
|
{onCancel && (
|
|
<button
|
|
type="button"
|
|
onClick={onCancel}
|
|
disabled={isLoading}
|
|
className="py-2 px-4 border border-gray-300 rounded-md text-sm font-medium text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500"
|
|
>
|
|
{t('common.cancel')}
|
|
</button>
|
|
)}
|
|
<button
|
|
type="submit"
|
|
disabled={isLoading}
|
|
className="py-2 px-4 border border-transparent rounded-md shadow-sm text-sm font-medium text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 btn-primary"
|
|
>
|
|
{isLoading ? (
|
|
<span className="flex items-center">
|
|
<svg className="animate-spin -ml-1 mr-2 h-4 w-4 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24">
|
|
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4"></circle>
|
|
<path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
|
</svg>
|
|
{t('common.save')}
|
|
</span>
|
|
) : (
|
|
t('common.save')
|
|
)}
|
|
</button>
|
|
</div>
|
|
</form>
|
|
)}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default ChangePasswordForm; |