2023-02-06 10:01:03 -08:00
|
|
|
import clsx from 'clsx';
|
2022-11-25 09:04:11 -08:00
|
|
|
import React from 'react';
|
2022-11-26 07:00:54 -08:00
|
|
|
import { defineMessages, FormattedMessage } from 'react-intl';
|
2022-04-12 06:52:04 -07:00
|
|
|
|
|
|
|
import { patchMe } from 'soapbox/actions/me';
|
|
|
|
import { Avatar, Button, Card, CardBody, Icon, Spinner, Stack, Text } from 'soapbox/components/ui';
|
2023-01-09 14:13:12 -08:00
|
|
|
import { useAppDispatch, useOwnAccount } from 'soapbox/hooks';
|
2022-12-20 08:34:53 -08:00
|
|
|
import toast from 'soapbox/toast';
|
2022-12-12 15:08:33 -08:00
|
|
|
import { isDefaultAvatar } from 'soapbox/utils/accounts';
|
2022-11-15 12:46:23 -08:00
|
|
|
import resizeImage from 'soapbox/utils/resize-image';
|
2022-04-12 06:52:04 -07:00
|
|
|
|
2022-06-09 11:56:14 -07:00
|
|
|
import type { AxiosError } from 'axios';
|
|
|
|
|
2022-11-26 07:00:54 -08:00
|
|
|
const messages = defineMessages({
|
|
|
|
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 AvatarSelectionStep = ({ onNext }: { onNext: () => void }) => {
|
2023-01-09 14:13:12 -08:00
|
|
|
const dispatch = useAppDispatch();
|
2023-06-25 10:35:09 -07:00
|
|
|
const { account } = useOwnAccount();
|
2022-04-12 06:52:04 -07:00
|
|
|
|
|
|
|
const fileInput = React.useRef<HTMLInputElement>(null);
|
|
|
|
const [selectedFile, setSelectedFile] = React.useState<string | null>();
|
|
|
|
const [isSubmitting, setSubmitting] = React.useState<boolean>(false);
|
|
|
|
const [isDisabled, setDisabled] = React.useState<boolean>(true);
|
2022-04-20 10:08:49 -07:00
|
|
|
const isDefault = account ? isDefaultAvatar(account.avatar) : false;
|
2022-04-12 06:52:04 -07:00
|
|
|
|
|
|
|
const openFilePicker = () => {
|
|
|
|
fileInput.current?.click();
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleFileChange = (event: React.ChangeEvent<HTMLInputElement>) => {
|
|
|
|
const maxPixels = 400 * 400;
|
2022-04-24 12:28:07 -07:00
|
|
|
const rawFile = event.target.files?.item(0);
|
|
|
|
|
|
|
|
if (!rawFile) return;
|
2022-04-12 06:52:04 -07:00
|
|
|
|
|
|
|
resizeImage(rawFile, maxPixels).then((file) => {
|
|
|
|
const url = file ? URL.createObjectURL(file) : account?.avatar as string;
|
|
|
|
|
|
|
|
setSelectedFile(url);
|
|
|
|
setSubmitting(true);
|
|
|
|
|
|
|
|
const formData = new FormData();
|
|
|
|
formData.append('avatar', rawFile);
|
|
|
|
const credentials = dispatch(patchMe(formData));
|
|
|
|
|
|
|
|
Promise.all([credentials]).then(() => {
|
|
|
|
setDisabled(false);
|
|
|
|
setSubmitting(false);
|
|
|
|
onNext();
|
|
|
|
}).catch((error: AxiosError) => {
|
|
|
|
setSubmitting(false);
|
|
|
|
setDisabled(false);
|
|
|
|
setSelectedFile(null);
|
|
|
|
|
|
|
|
if (error.response?.status === 422) {
|
2022-12-20 08:34:53 -08:00
|
|
|
toast.error((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
|
|
|
}
|
|
|
|
});
|
|
|
|
}).catch(console.error);
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Card variant='rounded' size='xl'>
|
|
|
|
<CardBody>
|
|
|
|
<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-900/50 sm:-mx-10 sm:pb-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.avatar.title' defaultMessage='Choose a profile picture' />
|
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.avatar.subtitle' defaultMessage='Just have fun with it.' />
|
2022-04-12 06:52:04 -07:00
|
|
|
</Text>
|
|
|
|
</Stack>
|
|
|
|
</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-12 06:52:04 -07:00
|
|
|
<Stack space={10}>
|
2023-02-01 14:13:42 -08:00
|
|
|
<div className='relative mx-auto rounded-full bg-gray-200'>
|
2022-04-12 06:52:04 -07:00
|
|
|
{account && (
|
|
|
|
<Avatar src={selectedFile || account.avatar} size={175} />
|
|
|
|
)}
|
|
|
|
|
|
|
|
{isSubmitting && (
|
2023-02-01 14:13:42 -08:00
|
|
|
<div className='absolute inset-0 flex items-center justify-center rounded-full bg-white/80 dark:bg-primary-900/80'>
|
2022-04-12 06:52:04 -07:00
|
|
|
<Spinner withText={false} />
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
|
|
|
|
<button
|
|
|
|
onClick={openFilePicker}
|
|
|
|
type='button'
|
2023-02-06 10:01:03 -08:00
|
|
|
className={clsx({
|
2022-07-22 10:30:16 -07:00
|
|
|
'absolute bottom-3 right-2 p-1 bg-primary-600 rounded-full ring-2 ring-white dark:ring-primary-900 hover:bg-primary-700': true,
|
2022-04-12 06:52:04 -07:00
|
|
|
'opacity-50 pointer-events-none': isSubmitting,
|
|
|
|
})}
|
|
|
|
disabled={isSubmitting}
|
|
|
|
>
|
2023-02-01 14:13:42 -08:00
|
|
|
<Icon src={require('@tabler/icons/plus.svg')} className='h-5 w-5 text-white' />
|
2022-04-12 06:52:04 -07:00
|
|
|
</button>
|
|
|
|
|
|
|
|
<input type='file' className='hidden' ref={fileInput} onChange={handleFileChange} />
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<Stack justifyContent='center' space={2}>
|
2022-04-20 10:08:49 -07:00
|
|
|
<Button block theme='primary' type='button' onClick={onNext} disabled={isDefault && isDisabled || isSubmitting}>
|
2022-04-12 11:25:53 -07:00
|
|
|
{isSubmitting ? (
|
2022-04-22 09:18:36 -07:00
|
|
|
<FormattedMessage id='onboarding.saving' defaultMessage='Saving…' />
|
2022-04-12 11:25:53 -07:00
|
|
|
) : (
|
|
|
|
<FormattedMessage id='onboarding.next' defaultMessage='Next' />
|
|
|
|
)}
|
2022-04-12 06:52:04 -07:00
|
|
|
</Button>
|
|
|
|
|
|
|
|
{isDisabled && (
|
2022-07-22 10:30:16 -07:00
|
|
|
<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 AvatarSelectionStep;
|