import classNames from 'clsx'; import React from 'react'; import { FormattedMessage } from 'react-intl'; import { useDispatch } from 'react-redux'; import { patchMe } from 'soapbox/actions/me'; import snackbar from 'soapbox/actions/snackbar'; import { Avatar, Button, Card, CardBody, Icon, Spinner, Stack, Text } from 'soapbox/components/ui'; import { useOwnAccount } from 'soapbox/hooks'; import resizeImage from 'soapbox/utils/resize-image'; import type { AxiosError } from 'axios'; /** Default avatar filenames from various backends */ const DEFAULT_AVATARS = [ '/avatars/original/missing.png', // Mastodon '/images/avi.png', // Pleroma ]; /** Check if the avatar is a default avatar */ const isDefaultAvatar = (url: string) => { return DEFAULT_AVATARS.every(avatar => url.endsWith(avatar)); }; const AvatarSelectionStep = ({ onNext }: { onNext: () => void }) => { const dispatch = useDispatch(); const account = useOwnAccount(); const fileInput = React.useRef(null); const [selectedFile, setSelectedFile] = React.useState(); const [isSubmitting, setSubmitting] = React.useState(false); const [isDisabled, setDisabled] = React.useState(true); const isDefault = account ? isDefaultAvatar(account.avatar) : false; const openFilePicker = () => { fileInput.current?.click(); }; const handleFileChange = (event: React.ChangeEvent) => { const maxPixels = 400 * 400; const rawFile = event.target.files?.item(0); if (!rawFile) return; 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) { dispatch(snackbar.error((error.response.data as any).error.replace('Validation failed: ', ''))); } else { dispatch(snackbar.error('An unexpected error occurred. Please try again or skip this step.')); } }); }).catch(console.error); }; return (
{account && ( )} {isSubmitting && (
)}
{isDisabled && ( )}
); }; export default AvatarSelectionStep;