bigbuffet-rw/app/soapbox/features/onboarding/steps/display-name-step.tsx

116 lines
4 KiB
TypeScript
Raw Normal View History

import React from 'react';
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
2022-04-12 06:52:04 -07:00
import { patchMe } from 'soapbox/actions/me';
import { Button, Card, CardBody, FormGroup, Input, Stack, Text } from 'soapbox/components/ui';
import { useAppDispatch, useOwnAccount } from 'soapbox/hooks';
2022-12-20 08:34:53 -08:00
import toast from 'soapbox/toast';
2022-04-12 06:52:04 -07:00
import type { AxiosError } from 'axios';
const messages = defineMessages({
usernamePlaceholder: { id: 'onboarding.display_name.placeholder', defaultMessage: 'Eg. John Smith' },
error: { id: 'onboarding.error', defaultMessage: 'An unexpected error occurred. Please try again or skip this step.' },
});
2022-04-12 06:52:04 -07:00
const DisplayNameStep = ({ onNext }: { onNext: () => void }) => {
const intl = useIntl();
const dispatch = useAppDispatch();
2022-04-12 06:52:04 -07:00
const account = useOwnAccount();
const [value, setValue] = React.useState<string>(account?.display_name || '');
2022-04-12 06:52:04 -07:00
const [isSubmitting, setSubmitting] = React.useState<boolean>(false);
const [errors, setErrors] = React.useState<string[]>([]);
const trimmedValue = value.trim();
const isValid = trimmedValue.length > 0;
const isDisabled = !isValid || value.length > 30;
const hintText = React.useMemo(() => {
const charsLeft = 30 - value.length;
const suffix = charsLeft === 1 ? 'character remaining' : 'characters remaining';
return `${charsLeft} ${suffix}`;
}, [value]);
const handleSubmit = () => {
setSubmitting(true);
const credentials = dispatch(patchMe({ display_name: value }));
Promise.all([credentials])
.then(() => {
setSubmitting(false);
onNext();
}).catch((error: AxiosError) => {
setSubmitting(false);
if (error.response?.status === 422) {
setErrors([(error.response.data as any).error.replace('Validation failed: ', '')]);
2022-04-12 06:52:04 -07:00
} else {
2022-12-20 08:34:53 -08:00
toast.error(messages.error);
2022-04-12 06:52:04 -07:00
}
});
};
return (
<Card variant='rounded' size='xl'>
<CardBody>
<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-04-12 06:52:04 -07:00
<Stack space={2}>
<Text size='2xl' align='center' weight='bold'>
2022-04-12 11:25:53 -07:00
<FormattedMessage id='onboarding.display_name.title' defaultMessage='Choose a display name' />
2022-04-12 06:52:04 -07:00
</Text>
<Text theme='muted' align='center'>
2022-04-12 11:25:53 -07:00
<FormattedMessage id='onboarding.display_name.subtitle' defaultMessage='You can always edit this later.' />
2022-04-12 06:52:04 -07:00
</Text>
</Stack>
</div>
<div className='sm:pt-10 sm:w-2/3 md:w-1/2 mx-auto'>
<Stack space={5}>
<FormGroup
hintText={hintText}
labelText={<FormattedMessage id='onboarding.display_name.label' defaultMessage='Display name' />}
2022-04-12 06:52:04 -07:00
errors={errors}
>
<Input
onChange={(event) => setValue(event.target.value)}
placeholder={intl.formatMessage(messages.usernamePlaceholder)}
2022-04-12 06:52:04 -07:00
type='text'
value={value}
maxLength={30}
/>
</FormGroup>
<Stack justifyContent='center' space={2}>
<Button
block
theme='primary'
type='submit'
disabled={isDisabled || isSubmitting}
onClick={handleSubmit}
>
{isSubmitting ? (
<FormattedMessage id='onboarding.saving' defaultMessage='Saving…' />
) : (
<FormattedMessage id='onboarding.next' defaultMessage='Next' />
)}
2022-04-12 06:52:04 -07:00
</Button>
<Button block theme='tertiary' type='button' onClick={onNext}>
2022-04-12 11:25:53 -07:00
<FormattedMessage id='onboarding.skip' defaultMessage='Skip for now' />
</Button>
2022-04-12 06:52:04 -07:00
</Stack>
</Stack>
</div>
</div>
</CardBody>
</Card>
);
};
export default DisplayNameStep;