diff --git a/examples/crm/src/authProvider.ts b/examples/crm/src/authProvider.ts index 7a6c1d09c6..0c18527106 100644 --- a/examples/crm/src/authProvider.ts +++ b/examples/crm/src/authProvider.ts @@ -6,6 +6,10 @@ export const DEFAULT_USER = { id: 0, first_name: 'Jane', last_name: 'Doe', + email: 'janedoe@atomic.dev', + password: 'demo', + administrator: true, + avatar: '', } as const; export const USER_STORAGE_KEY = 'user'; @@ -53,13 +57,12 @@ export const authProvider: AuthProvider = { getIdentity: () => { const userItem = localStorage.getItem(USER_STORAGE_KEY); const user = userItem ? (JSON.parse(userItem) as Sale) : null; - return Promise.resolve({ id: user?.id ?? 0, fullName: user ? `${user.first_name} ${user.last_name}` : 'Jane Doe', - avatar: '', + avatar: user?.avatar, }); }, }; diff --git a/examples/crm/src/contacts/ContactAvatarTest.tsx b/examples/crm/src/contacts/ContactAvatarTest.tsx deleted file mode 100644 index 7c6a359110..0000000000 --- a/examples/crm/src/contacts/ContactAvatarTest.tsx +++ /dev/null @@ -1,82 +0,0 @@ -import React, { createRef, useState } from 'react'; -import { Cropper, ReactCropperElement } from 'react-cropper'; -import 'cropperjs/dist/cropper.css'; -import { Button } from '@mui/material'; - -// this transforms file to base64 -const file2Base64 = (file: File): Promise => { - return new Promise((resolve, reject) => { - const reader = new FileReader(); - reader.readAsDataURL(file); - reader.onload = () => resolve(reader.result?.toString() || ''); - reader.onerror = error => reject(error); - }); -}; - -const ContactAvatarTest = () => { - // ref of the file input - const fileRef = createRef(); - - // the selected image - const [uploaded, setUploaded] = useState(null as string | null); - - // the resulting cropped image - const [cropped, setCropped] = useState(null as string | null); - - // the reference of cropper element - const cropperRef = createRef(); - - const onFileInputChange: React.ChangeEventHandler = e => { - e.preventDefault(); - e.stopPropagation(); - const file = e.target?.files?.[0]; - if (file) { - file2Base64(file).then(base64 => { - setUploaded(base64); - }); - } - }; - - const onCrop = () => { - const imageElement: any = cropperRef?.current; - const cropper: any = imageElement?.cropper; - setCropped(cropper.getCroppedCanvas().toDataURL()); - }; - - return ( - <> -
- {uploaded ? ( -
- - - {cropped && Cropped!} -
- ) : ( - <> - - - - )} -
- - ); -}; - -export default ContactAvatarTest; diff --git a/examples/crm/src/contacts/ContactInputs.tsx b/examples/crm/src/contacts/ContactInputs.tsx index fc6f5e45e1..534d5f21dc 100644 --- a/examples/crm/src/contacts/ContactInputs.tsx +++ b/examples/crm/src/contacts/ContactInputs.tsx @@ -17,25 +17,20 @@ import { useCreate, useGetIdentity, useNotify, - useRecordContext, } from 'react-admin'; import { isLinkedinUrl } from '../misc/isLinkedInUrl'; import { useConfigurationContext } from '../root/ConfigurationContext'; import { Sale } from '../types'; import { Avatar } from './Avatar'; -import AvatarEditor from 'react-avatar-editor'; -import ContactAvatarTest from './ContactAvatarTest'; export const ContactInputs = () => { const theme = useTheme(); const isMobile = useMediaQuery(theme.breakpoints.down('md')); - const record = useRecordContext(); return ( - { email: 'janedoe@atomic.dev', password: 'demo', administrator: true, + avatar: '', }, ...randomSales, ]; diff --git a/examples/crm/src/misc/ImageEditorField.tsx b/examples/crm/src/misc/ImageEditorField.tsx new file mode 100644 index 0000000000..7148f4a07a --- /dev/null +++ b/examples/crm/src/misc/ImageEditorField.tsx @@ -0,0 +1,145 @@ +import { + Avatar, + AvatarProps, + Box, + Button, + Dialog, + DialogActions, + DialogContent, + DialogTitle, + Stack, +} from '@mui/material'; +import 'cropperjs/dist/cropper.css'; +import { useFieldValue } from 'ra-core'; +import { createRef, useCallback, useState } from 'react'; +import { FieldProps, Toolbar } from 'react-admin'; +import { Cropper, ReactCropperElement } from 'react-cropper'; +import { useDropzone } from 'react-dropzone'; +import { useFormContext } from 'react-hook-form'; +import { DialogCloseButton } from './DialogCloseButton'; + +const ImageEditorField = (props: ImageEditorFieldProps) => { + const { getValues } = useFormContext(); + const imageUrl = getValues()[props.source]; + const [isHovered, setIsHovered] = useState(false); + const [isDialogOpen, setIsDialogOpen] = useState(false); + + if (!imageUrl) { + return null; + } + + const commonProps = { + src: imageUrl, + onMouseEnter: () => setIsHovered(true), + onMouseLeave: () => setIsHovered(false), + onClick: () => setIsDialogOpen(true), + style: { cursor: 'pointer' }, + sx: { + ...props.sx, + width: props.width || (props.type === 'avatar' ? 50 : 200), + height: props.height || (props.type === 'avatar' ? 50 : 200), + }, + }; + + return ( + <> + {props.type === 'avatar' ? ( + + ) : ( + + )} + setIsDialogOpen(false)} + {...props} + /> + + ); +}; + +const ImageEditorDialog = (props: ImageEditorDialogProps) => { + const { setValue, handleSubmit } = useFormContext(); + const cropperRef = createRef(); + const initialValue = useFieldValue(props); + const [imageSrc, setImageSrc] = useState(initialValue); + const onDrop = useCallback((files: File[]) => { + const preview = URL.createObjectURL(files[0]); + setImageSrc(preview); + }, []); + + const updateImage = () => { + const cropper = cropperRef.current?.cropper; + const croppedImage = cropper?.getCroppedCanvas().toDataURL(); + if (croppedImage) { + setImageSrc(croppedImage); + setValue(props.source, croppedImage); + props.onClose(); + + if (props.onSave) { + handleSubmit(props.onSave)(); + } + } + }; + + const { getRootProps, getInputProps } = useDropzone({ + accept: { 'image/jpeg': ['.jpeg', '.png'] }, + onDrop, + maxFiles: 1, + }); + + return ( + + + Resize your image + + + + +

Drop a file to upload, or click to select it.

+
+ +
+
+ + + + + +
+ ); +}; + +export default ImageEditorField; + +export interface ImageEditorFieldProps< + RecordType extends Record = Record, +> extends FieldProps, + AvatarProps { + width?: number; + height?: number; + type?: 'avatar' | 'image'; + onSave?: any; +} + +export interface ImageEditorDialogProps extends ImageEditorFieldProps { + open: boolean; + onClose: () => void; +} diff --git a/examples/crm/src/settings/SettingsPage.tsx b/examples/crm/src/settings/SettingsPage.tsx index a78bcf4224..bcdc414b5a 100644 --- a/examples/crm/src/settings/SettingsPage.tsx +++ b/examples/crm/src/settings/SettingsPage.tsx @@ -23,6 +23,7 @@ import { import { useFormState } from 'react-hook-form'; import { USER_STORAGE_KEY } from '../authProvider'; import { UpdatePassword } from './UpdatePassword'; +import ImageEditorField from '../misc/ImageEditorField'; export const SettingsPage = () => { const [update] = useUpdate(); @@ -49,7 +50,7 @@ export const SettingsPage = () => { JSON.stringify(data) ); refetch(); - setEditMode(true); + setEditMode(false); notify('Your profile has been updated'); }, onError: _ => { @@ -80,13 +81,46 @@ const SettingsForm = ({ isEditMode: boolean; setEditMode: (value: boolean) => void; }) => { + const [update] = useUpdate(); + const notify = useNotify(); + const { identity, refetch } = useGetIdentity(); const { isDirty } = useFormState(); const [openPasswordChange, setOpenPasswordChange] = useState(false); + if (!identity) return null; + const handleClickOpenPasswordChange = () => { setOpenPasswordChange(true); }; + const handleAvatarUpdate = async (values: any) => { + await update( + 'sales', + { + id: identity.id, + data: values, + previousData: identity, + }, + { + onSuccess: data => { + // Update local user + localStorage.setItem( + USER_STORAGE_KEY, + JSON.stringify(data) + ); + refetch(); + setEditMode(false); + notify('Your profile has been updated'); + }, + onError: _ => { + notify('An error occurred. Please try again', { + type: 'error', + }); + }, + } + ); + }; + return ( @@ -106,6 +140,11 @@ const SettingsForm = ({
+ @@ -121,7 +160,7 @@ const SettingsForm = ({ setOpen={setOpenPasswordChange} /> - {!isEditMode && ( + {isEditMode && (