mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-01-07 11:08:52 +01:00
135 lines
4.3 KiB
TypeScript
135 lines
4.3 KiB
TypeScript
|
|
import { useSearchParams, useNavigate } from 'react-router-dom';
|
||
|
|
import { useState, useEffect, useMemo, useCallback } from 'react';
|
||
|
|
import { useVerifyEmailMutation, useResendVerificationEmail } from '~/data-provider';
|
||
|
|
import { ThemeSelector } from '~/components/ui';
|
||
|
|
import { Spinner } from '~/components/svg';
|
||
|
|
import { useLocalize } from '~/hooks';
|
||
|
|
|
||
|
|
function RequestPasswordReset() {
|
||
|
|
const navigate = useNavigate();
|
||
|
|
const localize = useLocalize();
|
||
|
|
const [params] = useSearchParams();
|
||
|
|
|
||
|
|
const [countdown, setCountdown] = useState<number>(3);
|
||
|
|
const [headerText, setHeaderText] = useState<string>('');
|
||
|
|
const [showResendLink, setShowResendLink] = useState<boolean>(false);
|
||
|
|
const [verificationStatus, setVerificationStatus] = useState<boolean>(false);
|
||
|
|
|
||
|
|
const token = useMemo(() => params.get('token') || '', [params]);
|
||
|
|
const email = useMemo(() => params.get('email') || '', [params]);
|
||
|
|
|
||
|
|
const countdownRedirect = useCallback(() => {
|
||
|
|
setCountdown(3);
|
||
|
|
const timer = setInterval(() => {
|
||
|
|
setCountdown((prevCountdown) => {
|
||
|
|
if (prevCountdown <= 1) {
|
||
|
|
clearInterval(timer);
|
||
|
|
navigate('/c/new', { replace: true });
|
||
|
|
return 0;
|
||
|
|
} else {
|
||
|
|
return prevCountdown - 1;
|
||
|
|
}
|
||
|
|
});
|
||
|
|
}, 1000);
|
||
|
|
}, [navigate]);
|
||
|
|
|
||
|
|
const verifyEmailMutation = useVerifyEmailMutation({
|
||
|
|
onSuccess: () => {
|
||
|
|
setHeaderText(localize('com_auth_email_verification_success') + ' 🎉');
|
||
|
|
setVerificationStatus(true);
|
||
|
|
countdownRedirect();
|
||
|
|
},
|
||
|
|
onError: () => {
|
||
|
|
setShowResendLink(true);
|
||
|
|
setVerificationStatus(true);
|
||
|
|
setHeaderText(localize('com_auth_email_verification_failed') + ' 😢');
|
||
|
|
setCountdown(0);
|
||
|
|
},
|
||
|
|
});
|
||
|
|
|
||
|
|
const resendEmailMutation = useResendVerificationEmail({
|
||
|
|
onSuccess: () => {
|
||
|
|
setHeaderText(localize('com_auth_email_resent_success') + ' 📧');
|
||
|
|
countdownRedirect();
|
||
|
|
},
|
||
|
|
onError: () => {
|
||
|
|
setHeaderText(localize('com_auth_email_resent_failed') + ' 😢');
|
||
|
|
countdownRedirect();
|
||
|
|
},
|
||
|
|
onMutate: () => setShowResendLink(false),
|
||
|
|
});
|
||
|
|
|
||
|
|
const handleResendEmail = () => {
|
||
|
|
resendEmailMutation.mutate({ email });
|
||
|
|
};
|
||
|
|
|
||
|
|
useEffect(() => {
|
||
|
|
if (verifyEmailMutation.isLoading || verificationStatus) {
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (token && email) {
|
||
|
|
verifyEmailMutation.mutate({
|
||
|
|
email,
|
||
|
|
token,
|
||
|
|
});
|
||
|
|
return;
|
||
|
|
} else if (email) {
|
||
|
|
setHeaderText(localize('com_auth_email_verification_failed_token_missing') + ' 😢');
|
||
|
|
} else {
|
||
|
|
setHeaderText(localize('com_auth_email_verification_invalid') + ' 🤨');
|
||
|
|
}
|
||
|
|
|
||
|
|
setShowResendLink(true);
|
||
|
|
setVerificationStatus(true);
|
||
|
|
setCountdown(0);
|
||
|
|
}, [localize, token, email, verificationStatus, verifyEmailMutation]);
|
||
|
|
|
||
|
|
const VerificationSuccess = () => (
|
||
|
|
<div className="flex flex-col items-center justify-center">
|
||
|
|
<h1 className="mb-4 text-center text-3xl font-semibold text-black dark:text-white">
|
||
|
|
{headerText}
|
||
|
|
</h1>
|
||
|
|
{countdown > 0 && (
|
||
|
|
<p className="text-center text-lg text-gray-600 dark:text-gray-400">
|
||
|
|
{localize('com_auth_email_verification_redirecting', countdown.toString())}
|
||
|
|
</p>
|
||
|
|
)}
|
||
|
|
{showResendLink && countdown === 0 && (
|
||
|
|
<p className="text-center text-lg text-gray-600 dark:text-gray-400">
|
||
|
|
{localize('com_auth_email_verification_resend_prompt')}
|
||
|
|
<button
|
||
|
|
className="ml-2 text-blue-600 hover:underline"
|
||
|
|
onClick={handleResendEmail}
|
||
|
|
disabled={resendEmailMutation.isLoading}
|
||
|
|
>
|
||
|
|
{localize('com_auth_email_resend_link')}
|
||
|
|
</button>
|
||
|
|
</p>
|
||
|
|
)}
|
||
|
|
</div>
|
||
|
|
);
|
||
|
|
|
||
|
|
const VerificationInProgress = () => (
|
||
|
|
<div className="flex flex-col items-center justify-center">
|
||
|
|
<h1 className="mb-4 text-center text-3xl font-semibold text-black dark:text-white">
|
||
|
|
{localize('com_auth_email_verification_in_progress')}
|
||
|
|
</h1>
|
||
|
|
<div className="mt-4 flex justify-center">
|
||
|
|
<Spinner className="h-8 w-8 text-green-500" />
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
);
|
||
|
|
|
||
|
|
return (
|
||
|
|
<div className="flex min-h-screen flex-col items-center justify-center bg-white pt-6 dark:bg-gray-900 sm:pt-0">
|
||
|
|
<div className="absolute bottom-0 left-0 m-4">
|
||
|
|
<ThemeSelector />
|
||
|
|
</div>
|
||
|
|
{verificationStatus ? <VerificationSuccess /> : <VerificationInProgress />}
|
||
|
|
</div>
|
||
|
|
);
|
||
|
|
}
|
||
|
|
|
||
|
|
export default RequestPasswordReset;
|