| 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-500 dark:text-red-900"> | |
| {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 transition-colors hover:text-green-700 dark:text-green-400 dark:hover:text-green-300" | |
| > | |
| {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; | |