2022-06-13 08:34:59 -07:00
|
|
|
import { AxiosError } from 'axios';
|
2022-06-09 12:22:19 -07:00
|
|
|
import React from 'react';
|
2022-11-11 15:07:02 -08:00
|
|
|
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
2022-03-21 11:09:01 -07:00
|
|
|
import OtpInput from 'react-otp-input';
|
|
|
|
|
|
|
|
import { confirmPhoneVerification, requestPhoneVerification } from 'soapbox/actions/verification';
|
2022-07-13 07:42:58 -07:00
|
|
|
import { Button, Form, FormGroup, PhoneInput, Text } from 'soapbox/components/ui';
|
2022-06-09 12:22:19 -07:00
|
|
|
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
|
|
|
|
2022-11-11 15:07:02 -08:00
|
|
|
const messages = defineMessages({
|
|
|
|
verificationInvalid: { id: 'sms_verification.invalid', defaultMessage: 'Please enter a valid phone number.' },
|
|
|
|
verificationSuccess: { id: 'sms_verification.success', defaultMessage: 'A verification code has been sent to your phone number.' },
|
|
|
|
verificationFail: { id: 'sms_verification.fail', defaultMessage: 'Failed to send SMS message to your phone number.' },
|
|
|
|
verificationExpired: { id: 'sms_verification.expired', defaultMessage: 'Your SMS token has expired.' },
|
2022-11-26 05:09:28 -08:00
|
|
|
phoneLabel: { id: 'sms_verification.phone.label', defaultMessage: 'Phone number' },
|
2022-11-11 15:07:02 -08:00
|
|
|
});
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
const Statuses = {
|
|
|
|
IDLE: 'IDLE',
|
|
|
|
REQUESTED: 'REQUESTED',
|
|
|
|
FAIL: 'FAIL',
|
|
|
|
};
|
|
|
|
|
|
|
|
const SmsVerification = () => {
|
|
|
|
const intl = useIntl();
|
2022-06-09 12:22:19 -07:00
|
|
|
const dispatch = useAppDispatch();
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2022-06-20 06:46:43 -07:00
|
|
|
const isLoading = useAppSelector((state) => state.verification.isLoading) as boolean;
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2022-07-13 15:37:40 -07:00
|
|
|
const [phone, setPhone] = React.useState<string>();
|
2022-03-21 11:09:01 -07:00
|
|
|
const [status, setStatus] = React.useState(Statuses.IDLE);
|
|
|
|
const [verificationCode, setVerificationCode] = React.useState('');
|
|
|
|
const [requestedAnother, setAlreadyRequestedAnother] = React.useState(false);
|
|
|
|
|
2022-07-13 17:08:19 -07:00
|
|
|
const isValid = !!phone;
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2022-07-13 15:37:40 -07:00
|
|
|
const onChange = React.useCallback((phone?: string) => {
|
2022-07-13 07:42:58 -07:00
|
|
|
setPhone(phone);
|
2022-03-21 11:09:01 -07:00
|
|
|
}, []);
|
|
|
|
|
2023-01-10 15:03:15 -08:00
|
|
|
const handleSubmit: React.FormEventHandler = React.useCallback((event) => {
|
2022-03-21 11:09:01 -07:00
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
if (!isValid) {
|
|
|
|
setStatus(Statuses.IDLE);
|
2022-12-20 07:47:46 -08:00
|
|
|
toast.error(intl.formatMessage(messages.verificationInvalid));
|
2022-03-21 11:09:01 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-07-13 15:37:40 -07:00
|
|
|
dispatch(requestPhoneVerification(phone!)).then(() => {
|
2022-12-20 07:47:46 -08:00
|
|
|
toast.success(intl.formatMessage(messages.verificationSuccess));
|
2022-03-21 11:09:01 -07:00
|
|
|
setStatus(Statuses.REQUESTED);
|
2022-06-13 08:34:59 -07:00
|
|
|
}).catch((error: AxiosError) => {
|
2022-11-11 15:07:02 -08:00
|
|
|
const message = (error.response?.data as any)?.message || intl.formatMessage(messages.verificationFail);
|
2022-06-13 08:34:59 -07:00
|
|
|
|
2022-12-20 07:47:46 -08:00
|
|
|
toast.error(message);
|
2022-03-21 11:09:01 -07:00
|
|
|
setStatus(Statuses.FAIL);
|
|
|
|
});
|
|
|
|
}, [phone, isValid]);
|
|
|
|
|
2023-01-10 15:03:15 -08:00
|
|
|
const resendVerificationCode: React.MouseEventHandler = React.useCallback((event) => {
|
2022-03-21 11:09:01 -07:00
|
|
|
setAlreadyRequestedAnother(true);
|
|
|
|
handleSubmit(event);
|
|
|
|
}, [isValid]);
|
|
|
|
|
|
|
|
const submitVerification = () => {
|
|
|
|
// TODO: handle proper validation from Pepe -- expired vs invalid
|
|
|
|
dispatch(confirmPhoneVerification(verificationCode))
|
2022-12-20 07:47:46 -08:00
|
|
|
.catch(() => {
|
|
|
|
toast.error(intl.formatMessage(messages.verificationExpired));
|
|
|
|
});
|
2022-03-21 11:09:01 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
if (verificationCode.length === 6) {
|
|
|
|
submitVerification();
|
|
|
|
}
|
|
|
|
}, [verificationCode]);
|
|
|
|
|
|
|
|
if (status === Statuses.REQUESTED) {
|
|
|
|
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'>
|
2022-11-11 15:07:02 -08:00
|
|
|
<FormattedMessage id='sms_verification.sent.header' defaultMessage='Verification code' />
|
2022-03-21 11:09:01 -07:00
|
|
|
</h1>
|
|
|
|
</div>
|
|
|
|
|
2023-02-01 14:13:42 -08:00
|
|
|
<div className='mx-auto space-y-4 sm:w-2/3 sm:pt-10 md:w-1/2'>
|
2022-03-21 11:09:01 -07:00
|
|
|
<Text theme='muted' size='sm' align='center'>
|
2022-11-26 05:09:28 -08:00
|
|
|
<FormattedMessage id='sms_verification.sent.body' defaultMessage='We sent you a 6-digit code via SMS. Enter it below.' />
|
2022-03-21 11:09:01 -07:00
|
|
|
</Text>
|
|
|
|
|
|
|
|
<OtpInput
|
|
|
|
value={verificationCode}
|
|
|
|
onChange={setVerificationCode}
|
|
|
|
numInputs={6}
|
|
|
|
isInputNum
|
|
|
|
shouldAutoFocus
|
|
|
|
isDisabled={isLoading}
|
|
|
|
containerStyle='flex justify-center mt-2 space-x-4'
|
2022-07-22 10:30:16 -07:00
|
|
|
inputStyle='w-10i border-gray-300 dark:bg-gray-800 dark:border-gray-800 rounded-md focus:ring-2 focus:ring-indigo-500 focus:border-indigo-500'
|
2022-03-21 11:09:01 -07:00
|
|
|
/>
|
|
|
|
|
|
|
|
<div className='text-center'>
|
|
|
|
<Button
|
|
|
|
size='sm'
|
|
|
|
type='button'
|
2022-07-22 10:30:16 -07:00
|
|
|
theme='tertiary'
|
2022-03-21 11:09:01 -07:00
|
|
|
onClick={resendVerificationCode}
|
|
|
|
disabled={requestedAnother}
|
|
|
|
>
|
2022-11-26 05:09:28 -08:00
|
|
|
<FormattedMessage id='sms_verification.sent.actions.resend' defaultMessage='Resend verification code?' />
|
2022-03-21 11:09:01 -07:00
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
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'>
|
2022-11-11 15:07:02 -08:00
|
|
|
<FormattedMessage id='sms_verification.header' defaultMessage='Enter your phone number' />
|
|
|
|
</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}>
|
2022-11-26 05:09:28 -08:00
|
|
|
<FormGroup labelText={intl.formatMessage(messages.phoneLabel)}>
|
2022-07-13 07:42:58 -07:00
|
|
|
<PhoneInput
|
2022-03-21 11:09:01 -07:00
|
|
|
value={phone}
|
|
|
|
onChange={onChange}
|
|
|
|
required
|
|
|
|
/>
|
|
|
|
</FormGroup>
|
|
|
|
|
|
|
|
<div className='text-center'>
|
2022-11-26 05:09:28 -08:00
|
|
|
<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>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-07-13 17:08:19 -07:00
|
|
|
export { SmsVerification as default };
|