bigbuffet-rw/app/soapbox/features/verification/steps/email-verification.tsx

147 lines
4.9 KiB
TypeScript
Raw Normal View History

import { AxiosError } from 'axios';
import React from 'react';
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
2022-03-21 11:09:01 -07:00
import { checkEmailVerification, postEmailVerification, requestEmailVerification } from 'soapbox/actions/verification';
2022-03-21 11:09:01 -07:00
import Icon from 'soapbox/components/icon';
import { Button, Form, FormGroup, Input, Text } from 'soapbox/components/ui';
import { useAppDispatch, useAppSelector } from 'soapbox/hooks';
2022-12-20 07:47:46 -08:00
import toast from 'soapbox/toast';
2022-03-21 11:09:01 -07:00
const messages = defineMessages({
verificationSuccess: { id: 'email_verification.success', defaultMessage: 'Verification email sent successfully.' },
verificationFail: { id: 'email_verification.fail', defaultMessage: 'Failed to request email verification.' },
verificationFailTakenAlert: { id: 'email_verifilcation.exists', defaultMessage: 'This email has already been taken.' },
verificationFailTaken: { id: 'email_verification.taken', defaultMessage: 'is taken' },
emailLabel: { id: 'email_verification.email.label', defaultMessage: 'E-mail address' },
});
2022-03-21 11:09:01 -07:00
const Statuses = {
IDLE: 'IDLE',
REQUESTED: 'REQUESTED',
FAIL: 'FAIL',
};
const EMAIL_REGEX = /^[^@\s]+@[^@\s]+$/;
interface IEmailSent {
handleSubmit: React.FormEventHandler
}
const EmailSent: React.FC<IEmailSent> = ({ handleSubmit }) => {
const dispatch = useAppDispatch();
2022-03-21 11:09:01 -07:00
const checkEmailConfirmation = () => {
dispatch(checkEmailVerification())
.then(() => dispatch(postEmailVerification()))
.catch(() => null);
};
React.useEffect(() => {
const intervalId = setInterval(() => checkEmailConfirmation(), 2500);
return () => clearInterval(intervalId);
}, []);
return (
2023-02-01 14:13:42 -08:00
<div className='mx-auto flex flex-col items-center justify-center sm:pt-10'>
<Icon src={require('@tabler/icons/send.svg')} className='mb-5 h-12 w-12 text-primary-600 dark:text-primary-400' />
2022-03-21 11:09:01 -07:00
2023-02-01 14:13:42 -08:00
<div className='mb-4 space-y-2 text-center'>
2022-03-21 11:09:01 -07:00
<Text weight='bold' size='3xl'>We sent you an email</Text>
<Text theme='muted'>Click on the link in the email to validate your email.</Text>
</div>
<Button theme='tertiary' onClick={handleSubmit}>Resend verification email</Button>
2022-03-21 11:09:01 -07:00
</div>
);
};
const EmailVerification = () => {
const intl = useIntl();
const dispatch = useAppDispatch();
2022-03-21 11:09:01 -07:00
const isLoading = useAppSelector((state) => state.verification.isLoading) as boolean;
2022-03-21 11:09:01 -07:00
const [email, setEmail] = React.useState('');
const [status, setStatus] = React.useState(Statuses.IDLE);
const [errors, setErrors] = React.useState<Array<string>>([]);
2022-03-21 11:09:01 -07:00
const isValid = email.length > 0 && EMAIL_REGEX.test(email);
2023-01-10 15:03:15 -08:00
const onChange: React.ChangeEventHandler<HTMLInputElement> = React.useCallback((event) => {
setEmail(event.target.value);
}, []);
2022-03-21 11:09:01 -07:00
const handleSubmit: React.FormEventHandler = React.useCallback((event) => {
2022-03-21 11:09:01 -07:00
event.preventDefault();
setErrors([]);
submitEmailForVerification();
}, [email]);
const submitEmailForVerification = () => {
return dispatch(requestEmailVerification((email)))
.then(() => {
setStatus(Statuses.REQUESTED);
2022-12-20 07:47:46 -08:00
toast.success(intl.formatMessage(messages.verificationSuccess));
2022-03-21 11:09:01 -07:00
})
.catch((error: AxiosError) => {
2022-08-16 06:48:58 -07:00
const errorMessage = (error.response?.data as any)?.error;
const isEmailTaken = errorMessage === 'email_taken';
let message = intl.formatMessage(messages.verificationFail);
2022-03-21 11:09:01 -07:00
2022-08-16 06:48:58 -07:00
if (isEmailTaken) {
message = intl.formatMessage(messages.verificationFailTakenAlert);
2022-08-16 06:48:58 -07:00
} else if (errorMessage) {
message = errorMessage;
}
2022-03-21 11:09:01 -07:00
if (isEmailTaken) {
setErrors([intl.formatMessage(messages.verificationFailTaken)]);
2022-03-21 11:09:01 -07:00
}
2022-12-20 07:47:46 -08:00
toast.error(message);
2022-03-21 11:09:01 -07:00
setStatus(Statuses.FAIL);
});
};
if (status === Statuses.REQUESTED) {
return <EmailSent handleSubmit={handleSubmit} />;
}
return (
<div>
2023-02-01 14:13:42 -08:00
<div className='-mx-4 mb-4 border-b border-solid border-gray-200 pb-4 dark:border-gray-800 sm:-mx-10 sm:pb-10'>
<h1 className='text-center text-2xl font-bold'>
<FormattedMessage id='email_verification.header' defaultMessage='Enter your email address' />
</h1>
2022-03-21 11:09:01 -07:00
</div>
2023-02-01 14:13:42 -08:00
<div className='mx-auto sm:w-2/3 sm:pt-10 md:w-1/2'>
2022-04-04 08:54:00 -07:00
<Form onSubmit={handleSubmit}>
<FormGroup labelText={intl.formatMessage(messages.emailLabel)} errors={errors}>
2022-03-21 11:09:01 -07:00
<Input
type='email'
value={email}
name='email'
onChange={onChange}
placeholder='you@email.com'
required
/>
</FormGroup>
<div className='text-center'>
<Button block theme='primary' type='submit' disabled={isLoading || !isValid}>
<FormattedMessage id='onboarding.next' defaultMessage='Next' />
</Button>
2022-03-21 11:09:01 -07:00
</div>
</Form>
</div>
</div>
);
};
export default EmailVerification;