bigbuffet-rw/app/soapbox/features/verification/registration.tsx

180 lines
5.9 KiB
TypeScript
Raw Normal View History

2022-03-21 11:09:01 -07:00
import * as React from 'react';
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
2022-03-21 11:09:01 -07:00
import { Redirect } from 'react-router-dom';
import { logIn, verifyCredentials } from 'soapbox/actions/auth';
import { fetchInstance } from 'soapbox/actions/instance';
2022-05-02 13:55:52 -07:00
import { startOnboarding } from 'soapbox/actions/onboarding';
2022-03-21 11:09:01 -07:00
import snackbar from 'soapbox/actions/snackbar';
import { createAccount, removeStoredVerification } from 'soapbox/actions/verification';
2022-07-25 12:41:23 -07:00
import { Button, Form, FormGroup, Input, Text } from 'soapbox/components/ui';
import { useAppDispatch, useAppSelector, useSoapboxConfig } from 'soapbox/hooks';
import { getRedirectUrl } from 'soapbox/utils/redirect';
2022-03-21 11:09:01 -07:00
2022-06-09 12:51:50 -07:00
import PasswordIndicator from './components/password-indicator';
import type { AxiosError } from 'axios';
const messages = defineMessages({
success: {
id: 'registrations.success',
defaultMessage: 'Welcome to {siteTitle}!',
},
usernameHint: {
id: 'registrations.username.hint',
defaultMessage: 'May only contain A-Z, 0-9, and underscores',
},
usernameTaken: {
id: 'registrations.unprocessable_entity',
defaultMessage: 'This username has already been taken.',
},
error: {
id: 'registrations.error',
defaultMessage: 'Failed to register your account.',
},
});
2022-03-21 11:09:01 -07:00
const initialState = {
username: '',
password: '',
};
const Registration = () => {
const dispatch = useAppDispatch();
2022-03-21 11:09:01 -07:00
const intl = useIntl();
2022-07-25 12:41:23 -07:00
const soapboxConfig = useSoapboxConfig();
const { links } = soapboxConfig;
2022-03-21 11:09:01 -07:00
const isLoading = useAppSelector((state) => state.verification.isLoading as boolean);
2022-04-07 10:39:22 -07:00
const siteTitle = useAppSelector((state) => state.instance.title);
2022-03-21 11:09:01 -07:00
const [state, setState] = React.useState(initialState);
2022-04-07 10:39:22 -07:00
const [shouldRedirect, setShouldRedirect] = React.useState<boolean>(false);
2022-06-09 12:51:50 -07:00
const [hasValidPassword, setHasValidPassword] = React.useState<boolean>(false);
2022-03-21 11:09:01 -07:00
const { username, password } = state;
const handleSubmit = React.useCallback((event) => {
event.preventDefault();
dispatch(createAccount(username, password))
.then(() => dispatch(logIn(username, password)))
2022-04-07 10:39:22 -07:00
.then(({ access_token }: any) => dispatch(verifyCredentials(access_token)))
2022-03-21 11:09:01 -07:00
.then(() => dispatch(fetchInstance()))
.then(() => {
setShouldRedirect(true);
removeStoredVerification();
2022-05-02 13:55:52 -07:00
dispatch(startOnboarding());
2022-03-21 11:09:01 -07:00
dispatch(
snackbar.success(
intl.formatMessage(messages.success, { siteTitle }),
2022-03-21 11:09:01 -07:00
),
);
})
2022-04-07 10:39:22 -07:00
.catch((error: AxiosError) => {
2022-03-21 11:09:01 -07:00
if (error?.response?.status === 422) {
dispatch(
snackbar.error(
intl.formatMessage(messages.usernameTaken),
2022-03-21 11:09:01 -07:00
),
);
} else {
dispatch(
snackbar.error(
intl.formatMessage(messages.error),
2022-03-21 11:09:01 -07:00
),
);
}
});
}, [username, password]);
const handleInputChange = React.useCallback((event) => {
event.persist();
setState((prevState) => ({ ...prevState, [event.target.name]: event.target.value }));
}, []);
if (shouldRedirect) {
const redirectUri = getRedirectUrl();
return <Redirect to={redirectUri} />;
2022-03-21 11:09:01 -07:00
}
return (
<div>
<div className='pb-4 sm:pb-10 mb-4 border-b border-gray-200 dark:border-gray-800 border-solid -mx-4 sm:-mx-10'>
2022-03-21 11:09:01 -07:00
<h1 className='text-center font-bold text-2xl'>
<FormattedMessage id='registration.header' defaultMessage='Register your account' />
2022-03-21 11:09:01 -07:00
</h1>
</div>
<div className='sm:pt-10 sm:w-2/3 md:w-1/2 mx-auto space-y-4'>
2022-04-04 08:54:00 -07:00
<Form onSubmit={handleSubmit}>
2022-10-13 10:26:34 -07:00
<FormGroup labelText='Your username' hintText={intl.formatMessage(messages.usernameHint)}>
2022-03-21 11:09:01 -07:00
<Input
name='username'
type='text'
value={username}
onChange={handleInputChange}
required
icon={require('@tabler/icons/at.svg')}
placeholder='LibertyForAll'
2022-03-21 11:09:01 -07:00
/>
</FormGroup>
<FormGroup labelText='Password'>
<Input
name='password'
type='password'
value={password}
onChange={handleInputChange}
required
data-testid='password-input'
2022-03-21 11:09:01 -07:00
/>
2022-06-09 08:03:12 -07:00
2022-06-09 12:51:50 -07:00
<PasswordIndicator password={password} onChange={setHasValidPassword} />
2022-03-21 11:09:01 -07:00
</FormGroup>
2022-07-25 12:41:23 -07:00
<div className='text-center space-y-2'>
<Button
block
theme='primary'
type='submit'
disabled={isLoading || !hasValidPassword}
>
Register
</Button>
2022-07-25 12:41:23 -07:00
{(links.get('termsOfService') && links.get('privacyPolicy')) ? (
<Text theme='muted' size='xs'>
<FormattedMessage
id='registration.acceptance'
2022-07-26 11:51:12 -07:00
defaultMessage='By registering, you agree to the {terms} and {privacy}.'
2022-07-25 12:41:23 -07:00
values={{
terms: (
2022-07-26 11:51:12 -07:00
<a href={links.get('termsOfService')} target='_blank' className='text-primary-600 dark:text-primary-400 hover:underline'>
2022-07-25 12:41:23 -07:00
<FormattedMessage
id='registration.tos'
defaultMessage='Terms of Service'
/>
</a>
),
privacy: (
2022-07-26 11:51:12 -07:00
<a href={links.get('privacyPolicy')} target='_blank' className='text-primary-600 dark:text-primary-400 hover:underline'>
2022-07-25 12:41:23 -07:00
<FormattedMessage
id='registration.privacy'
defaultMessage='Privacy Policy'
/>
</a>
),
}}
/>
</Text>
) : null}
2022-03-21 11:09:01 -07:00
</div>
</Form>
</div>
</div>
);
};
export default Registration;