mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-30 23:28:52 +01:00
* feat: do not open login footer links in new tab * feat: underline login links on hover for better accessibility * feat: nicer visuals for links on hover and focus
188 lines
7.6 KiB
TypeScript
188 lines
7.6 KiB
TypeScript
import React, { useState, useEffect, useContext } from 'react';
|
|
import { useForm } from 'react-hook-form';
|
|
import { Turnstile } from '@marsidev/react-turnstile';
|
|
import { ThemeContext, Spinner, Button, isDark } from '@librechat/client';
|
|
import type { TLoginUser, TStartupConfig } from 'librechat-data-provider';
|
|
import type { TAuthContext } from '~/common';
|
|
import { useResendVerificationEmail, useGetStartupConfig } from '~/data-provider';
|
|
import { useLocalize } from '~/hooks';
|
|
|
|
type TLoginFormProps = {
|
|
onSubmit: (data: TLoginUser) => void;
|
|
startupConfig: TStartupConfig;
|
|
error: Pick<TAuthContext, 'error'>['error'];
|
|
setError: Pick<TAuthContext, 'setError'>['setError'];
|
|
};
|
|
|
|
const LoginForm: React.FC<TLoginFormProps> = ({ onSubmit, startupConfig, error, setError }) => {
|
|
const localize = useLocalize();
|
|
const { theme } = useContext(ThemeContext);
|
|
const {
|
|
register,
|
|
getValues,
|
|
handleSubmit,
|
|
formState: { errors, isSubmitting },
|
|
} = useForm<TLoginUser>();
|
|
const [showResendLink, setShowResendLink] = useState<boolean>(false);
|
|
const [turnstileToken, setTurnstileToken] = useState<string | null>(null);
|
|
|
|
const { data: config } = useGetStartupConfig();
|
|
const useUsernameLogin = config?.ldap?.username;
|
|
const validTheme = isDark(theme) ? 'dark' : 'light';
|
|
const requireCaptcha = Boolean(startupConfig.turnstile?.siteKey);
|
|
|
|
useEffect(() => {
|
|
if (error && error.includes('422') && !showResendLink) {
|
|
setShowResendLink(true);
|
|
}
|
|
}, [error, showResendLink]);
|
|
|
|
const resendLinkMutation = useResendVerificationEmail({
|
|
onMutate: () => {
|
|
setError(undefined);
|
|
setShowResendLink(false);
|
|
},
|
|
});
|
|
|
|
if (!startupConfig) {
|
|
return null;
|
|
}
|
|
|
|
const renderError = (fieldName: string) => {
|
|
const errorMessage = errors[fieldName]?.message;
|
|
return errorMessage ? (
|
|
<span role="alert" className="mt-1 text-sm text-red-600 dark:text-red-500">
|
|
{String(errorMessage)}
|
|
</span>
|
|
) : null;
|
|
};
|
|
|
|
const handleResendEmail = () => {
|
|
const email = getValues('email');
|
|
if (!email) {
|
|
return setShowResendLink(false);
|
|
}
|
|
resendLinkMutation.mutate({ email });
|
|
};
|
|
|
|
return (
|
|
<>
|
|
{showResendLink && (
|
|
<div className="mt-2 rounded-md border border-green-500 bg-green-500/10 px-3 py-2 text-sm text-gray-600 dark:text-gray-200">
|
|
{localize('com_auth_email_verification_resend_prompt')}
|
|
<button
|
|
type="button"
|
|
className="ml-2 text-blue-600 hover:underline"
|
|
onClick={handleResendEmail}
|
|
disabled={resendLinkMutation.isLoading}
|
|
>
|
|
{localize('com_auth_email_resend_link')}
|
|
</button>
|
|
</div>
|
|
)}
|
|
<form
|
|
className="mt-6"
|
|
aria-label="Login form"
|
|
method="POST"
|
|
onSubmit={handleSubmit((data) => onSubmit(data))}
|
|
>
|
|
<div className="mb-4">
|
|
<div className="relative">
|
|
<input
|
|
type="text"
|
|
id="email"
|
|
autoComplete={useUsernameLogin ? 'username' : 'email'}
|
|
aria-label={localize('com_auth_email')}
|
|
{...register('email', {
|
|
required: localize('com_auth_email_required'),
|
|
maxLength: { value: 120, message: localize('com_auth_email_max_length') },
|
|
pattern: {
|
|
value: useUsernameLogin ? /\S+/ : /\S+@\S+\.\S+/,
|
|
message: localize('com_auth_email_pattern'),
|
|
},
|
|
})}
|
|
aria-invalid={!!errors.email}
|
|
className="webkit-dark-styles transition-color peer w-full rounded-2xl border border-border-light bg-surface-primary px-3.5 pb-2.5 pt-3 text-text-primary duration-200 focus:border-green-500 focus:outline-none"
|
|
placeholder=" "
|
|
/>
|
|
<label
|
|
htmlFor="email"
|
|
className="absolute start-3 top-1.5 z-10 origin-[0] -translate-y-4 scale-75 transform bg-surface-primary px-2 text-sm text-text-secondary-alt duration-200 peer-placeholder-shown:top-1/2 peer-placeholder-shown:-translate-y-1/2 peer-placeholder-shown:scale-100 peer-focus:top-1.5 peer-focus:-translate-y-4 peer-focus:scale-75 peer-focus:px-2 peer-focus:text-green-600 dark:peer-focus:text-green-500 rtl:peer-focus:left-auto rtl:peer-focus:translate-x-1/4"
|
|
>
|
|
{useUsernameLogin
|
|
? localize('com_auth_username').replace(/ \(.*$/, '')
|
|
: localize('com_auth_email_address')}
|
|
</label>
|
|
</div>
|
|
{renderError('email')}
|
|
</div>
|
|
<div className="mb-2">
|
|
<div className="relative">
|
|
<input
|
|
type="password"
|
|
id="password"
|
|
autoComplete="current-password"
|
|
aria-label={localize('com_auth_password')}
|
|
{...register('password', {
|
|
required: localize('com_auth_password_required'),
|
|
minLength: {
|
|
value: startupConfig?.minPasswordLength || 8,
|
|
message: localize('com_auth_password_min_length'),
|
|
},
|
|
maxLength: { value: 128, message: localize('com_auth_password_max_length') },
|
|
})}
|
|
aria-invalid={!!errors.password}
|
|
className="webkit-dark-styles transition-color peer w-full rounded-2xl border border-border-light bg-surface-primary px-3.5 pb-2.5 pt-3 text-text-primary duration-200 focus:border-green-500 focus:outline-none"
|
|
placeholder=" "
|
|
/>
|
|
<label
|
|
htmlFor="password"
|
|
className="absolute start-3 top-1.5 z-10 origin-[0] -translate-y-4 scale-75 transform bg-surface-primary px-2 text-sm text-text-secondary-alt duration-200 peer-placeholder-shown:top-1/2 peer-placeholder-shown:-translate-y-1/2 peer-placeholder-shown:scale-100 peer-focus:top-1.5 peer-focus:-translate-y-4 peer-focus:scale-75 peer-focus:px-2 peer-focus:text-green-600 dark:peer-focus:text-green-500 rtl:peer-focus:left-auto rtl:peer-focus:translate-x-1/4"
|
|
>
|
|
{localize('com_auth_password')}
|
|
</label>
|
|
</div>
|
|
{renderError('password')}
|
|
</div>
|
|
{startupConfig.passwordResetEnabled && (
|
|
<a
|
|
href="/forgot-password"
|
|
className="inline-flex p-1 text-sm font-medium text-green-600 underline decoration-transparent transition-all duration-200 hover:text-green-700 hover:decoration-green-700 focus:text-green-700 focus:decoration-green-700 dark:text-green-500 dark:hover:text-green-400 dark:hover:decoration-green-400 dark:focus:text-green-400 dark:focus:decoration-green-400"
|
|
>
|
|
{localize('com_auth_password_forgot')}
|
|
</a>
|
|
)}
|
|
|
|
{requireCaptcha && (
|
|
<div className="my-4 flex justify-center">
|
|
<Turnstile
|
|
siteKey={startupConfig.turnstile!.siteKey}
|
|
options={{
|
|
...startupConfig.turnstile!.options,
|
|
theme: validTheme,
|
|
}}
|
|
onSuccess={setTurnstileToken}
|
|
onError={() => setTurnstileToken(null)}
|
|
onExpire={() => setTurnstileToken(null)}
|
|
/>
|
|
</div>
|
|
)}
|
|
|
|
<div className="mt-6">
|
|
<Button
|
|
aria-label={localize('com_auth_continue')}
|
|
data-testid="login-button"
|
|
type="submit"
|
|
disabled={(requireCaptcha && !turnstileToken) || isSubmitting}
|
|
variant="submit"
|
|
className="h-12 w-full rounded-2xl"
|
|
>
|
|
{isSubmitting ? <Spinner /> : localize('com_auth_continue')}
|
|
</Button>
|
|
</div>
|
|
</form>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default LoginForm;
|