| import { useForm } from 'react-hook-form'; |
| import { useState, ReactNode } from 'react'; |
| import { Spinner, Button } from '@librechat/client'; |
| import { useOutletContext } from 'react-router-dom'; |
| import { useRequestPasswordResetMutation } from 'librechat-data-provider/react-query'; |
| import { loginPage } from 'librechat-data-provider'; |
| import type { TRequestPasswordReset, TRequestPasswordResetResponse } from 'librechat-data-provider'; |
| import type { TLoginLayoutContext } from '~/common'; |
| import type { FC } from 'react'; |
| import { useLocalize } from '~/hooks'; |
|
|
| const BodyTextWrapper: FC<{ children: ReactNode }> = ({ children }) => { |
| return ( |
| <div |
| className="relative mt-6 rounded-xl border border-green-500/20 bg-green-50/50 px-6 py-4 text-green-700 shadow-sm transition-all dark:bg-green-950/30 dark:text-green-100" |
| role="alert" |
| > |
| {children} |
| </div> |
| ); |
| }; |
|
|
| const ResetPasswordBodyText = () => { |
| const localize = useLocalize(); |
| return ( |
| <div className="flex flex-col space-y-4"> |
| <p>{localize('com_auth_reset_password_if_email_exists')}</p> |
| <a |
| className="inline-flex text-sm font-medium text-green-600 transition-colors hover:text-green-700 dark:text-green-400 dark:hover:text-green-300" |
| href={loginPage()} |
| > |
| {localize('com_auth_back_to_login')} |
| </a> |
| </div> |
| ); |
| }; |
|
|
| function RequestPasswordReset() { |
| const localize = useLocalize(); |
| const { |
| register, |
| handleSubmit, |
| formState: { errors }, |
| } = useForm<TRequestPasswordReset>(); |
| const [bodyText, setBodyText] = useState<ReactNode | undefined>(undefined); |
| const { startupConfig, setHeaderText } = useOutletContext<TLoginLayoutContext>(); |
|
|
| const requestPasswordReset = useRequestPasswordResetMutation(); |
| const { isLoading } = requestPasswordReset; |
|
|
| const onSubmit = (data: TRequestPasswordReset) => { |
| requestPasswordReset.mutate(data, { |
| onSuccess: (data: TRequestPasswordResetResponse) => { |
| if (data.link && !startupConfig?.emailEnabled) { |
| setHeaderText('com_auth_reset_password'); |
| setBodyText( |
| <span> |
| {localize('com_auth_click')}{' '} |
| <a className="text-green-500 hover:underline" href={data.link}> |
| {localize('com_auth_here')} |
| </a>{' '} |
| {localize('com_auth_to_reset_your_password')} |
| </span>, |
| ); |
| } else { |
| setHeaderText('com_auth_reset_password_link_sent'); |
| setBodyText(<ResetPasswordBodyText />); |
| } |
| }, |
| onError: () => { |
| setHeaderText('com_auth_reset_password_link_sent'); |
| setBodyText(<ResetPasswordBodyText />); |
| }, |
| }); |
| }; |
|
|
| if (bodyText) { |
| return <BodyTextWrapper>{bodyText}</BodyTextWrapper>; |
| } |
|
|
| return ( |
| <form |
| className="mt-8 space-y-6" |
| aria-label="Password reset form" |
| method="POST" |
| onSubmit={handleSubmit(onSubmit)} |
| > |
| <div className="space-y-2"> |
| <div className="relative"> |
| <input |
| type="email" |
| id="email" |
| autoComplete="off" |
| aria-label={localize('com_auth_email')} |
| {...register('email', { |
| required: localize('com_auth_email_required'), |
| minLength: { |
| value: 3, |
| message: localize('com_auth_email_min_length'), |
| }, |
| maxLength: { |
| value: 120, |
| message: localize('com_auth_email_max_length'), |
| }, |
| pattern: { |
| value: /\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 -top-2 left-2 z-10 bg-white px-2 text-sm text-gray-600 transition-all peer-placeholder-shown:top-3 peer-placeholder-shown:text-base peer-placeholder-shown:text-gray-500 peer-focus:-top-2 peer-focus:text-sm peer-focus:text-green-600 dark:bg-gray-900 dark:text-gray-400 dark:peer-focus:text-green-500" |
| > |
| {localize('com_auth_email_address')} |
| </label> |
| </div> |
| {errors.email && ( |
| <p role="alert" className="text-sm font-medium text-red-600 dark:text-red-400"> |
| {errors.email.message} |
| </p> |
| )} |
| </div> |
| <div className="space-y-4"> |
| <Button |
| aria-label="Continue with password reset" |
| type="submit" |
| disabled={!!errors.email || isLoading} |
| variant="submit" |
| className="h-12 w-full rounded-2xl" |
| > |
| {isLoading ? <Spinner /> : localize('com_auth_continue')} |
| </Button> |
| <a |
| href={loginPage()} |
| className="block text-center 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_back_to_login')} |
| </a> |
| </div> |
| </form> |
| ); |
| } |
|
|
| export default RequestPasswordReset; |
|
|