169 lines
6.1 KiB
TypeScript
169 lines
6.1 KiB
TypeScript
import React from 'react';
|
|
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
|
import { Redirect } from 'react-router-dom';
|
|
|
|
import { logIn, verifyCredentials } from 'soapbox/actions/auth';
|
|
import { fetchInstance } from 'soapbox/actions/instance';
|
|
import { startOnboarding } from 'soapbox/actions/onboarding';
|
|
import snackbar from 'soapbox/actions/snackbar';
|
|
import { createAccount, removeStoredVerification } from 'soapbox/actions/verification';
|
|
import { Button, Form, FormGroup, Input, Text } from 'soapbox/components/ui';
|
|
import { useAppDispatch, useAppSelector, useInstance, useSoapboxConfig } from 'soapbox/hooks';
|
|
import { getRedirectUrl } from 'soapbox/utils/redirect';
|
|
|
|
import PasswordIndicator from './components/password-indicator';
|
|
|
|
import type { AxiosError } from 'axios';
|
|
|
|
const messages = defineMessages({
|
|
success: { id: 'registrations.success', defaultMessage: 'Welcome to {siteTitle}!' },
|
|
usernameLabel: { id: 'registrations.username.label', defaultMessage: 'Your username' },
|
|
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.' },
|
|
passwordLabel: { id: 'registrations.password.label', defaultMessage: 'Password' },
|
|
error: { id: 'registrations.error', defaultMessage: 'Failed to register your account.' },
|
|
});
|
|
|
|
const initialState = {
|
|
username: '',
|
|
password: '',
|
|
};
|
|
|
|
const Registration = () => {
|
|
const dispatch = useAppDispatch();
|
|
const intl = useIntl();
|
|
const instance = useInstance();
|
|
const soapboxConfig = useSoapboxConfig();
|
|
const { links } = soapboxConfig;
|
|
|
|
const isLoading = useAppSelector((state) => state.verification.isLoading as boolean);
|
|
|
|
const [state, setState] = React.useState(initialState);
|
|
const [shouldRedirect, setShouldRedirect] = React.useState<boolean>(false);
|
|
const [hasValidPassword, setHasValidPassword] = React.useState<boolean>(false);
|
|
const { username, password } = state;
|
|
|
|
const handleSubmit = React.useCallback((event) => {
|
|
event.preventDefault();
|
|
|
|
dispatch(createAccount(username, password))
|
|
.then(() => dispatch(logIn(username, password)))
|
|
.then(({ access_token }: any) => dispatch(verifyCredentials(access_token)))
|
|
.then(() => dispatch(fetchInstance()))
|
|
.then(() => {
|
|
setShouldRedirect(true);
|
|
removeStoredVerification();
|
|
dispatch(startOnboarding());
|
|
dispatch(
|
|
snackbar.success(
|
|
intl.formatMessage(messages.success, { siteTitle: instance.title }),
|
|
),
|
|
);
|
|
})
|
|
.catch((error: AxiosError) => {
|
|
if (error?.response?.status === 422) {
|
|
dispatch(
|
|
snackbar.error(
|
|
intl.formatMessage(messages.usernameTaken),
|
|
),
|
|
);
|
|
} else {
|
|
dispatch(
|
|
snackbar.error(
|
|
intl.formatMessage(messages.error),
|
|
),
|
|
);
|
|
}
|
|
});
|
|
}, [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} />;
|
|
}
|
|
|
|
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'>
|
|
<h1 className='text-center font-bold text-2xl'>
|
|
<FormattedMessage id='registration.header' defaultMessage='Register your account' />
|
|
</h1>
|
|
</div>
|
|
|
|
<div className='sm:pt-10 sm:w-2/3 md:w-1/2 mx-auto space-y-4'>
|
|
<Form onSubmit={handleSubmit}>
|
|
<FormGroup labelText={intl.formatMessage(messages.usernameLabel)} hintText={intl.formatMessage(messages.usernameHint)}>
|
|
<Input
|
|
name='username'
|
|
type='text'
|
|
value={username}
|
|
onChange={handleInputChange}
|
|
required
|
|
icon={require('@tabler/icons/at.svg')}
|
|
placeholder='LibertyForAll'
|
|
/>
|
|
</FormGroup>
|
|
|
|
<FormGroup labelText={intl.formatMessage(messages.passwordLabel)}>
|
|
<Input
|
|
name='password'
|
|
type='password'
|
|
value={password}
|
|
onChange={handleInputChange}
|
|
required
|
|
data-testid='password-input'
|
|
/>
|
|
|
|
<PasswordIndicator password={password} onChange={setHasValidPassword} />
|
|
</FormGroup>
|
|
|
|
<div className='text-center space-y-2'>
|
|
<Button
|
|
block
|
|
theme='primary'
|
|
type='submit'
|
|
disabled={isLoading || !hasValidPassword}
|
|
>
|
|
<FormattedMessage id='header.register.label' defaultMessage='Register' />
|
|
</Button>
|
|
|
|
{(links.get('termsOfService') && links.get('privacyPolicy')) ? (
|
|
<Text theme='muted' size='xs'>
|
|
<FormattedMessage
|
|
id='registration.acceptance'
|
|
defaultMessage='By registering, you agree to the {terms} and {privacy}.'
|
|
values={{
|
|
terms: (
|
|
<a href={links.get('termsOfService')} target='_blank' className='text-primary-600 dark:text-primary-400 hover:underline'>
|
|
<FormattedMessage
|
|
id='registration.tos'
|
|
defaultMessage='Terms of Service'
|
|
/>
|
|
</a>
|
|
),
|
|
privacy: (
|
|
<a href={links.get('privacyPolicy')} target='_blank' className='text-primary-600 dark:text-primary-400 hover:underline'>
|
|
<FormattedMessage
|
|
id='registration.privacy'
|
|
defaultMessage='Privacy Policy'
|
|
/>
|
|
</a>
|
|
),
|
|
}}
|
|
/>
|
|
</Text>
|
|
) : null}
|
|
</div>
|
|
</Form>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default Registration;
|