import { useCallback, ReactNode, useState, useEffect, KeyboardEvent, } from 'react'; import { useAppDispatch, useAppSelector } from 'app/storeHooks'; import { FileRejection, useDropzone } from 'react-dropzone'; import { useToast } from '@chakra-ui/react'; import { ImageUploaderTriggerContext } from 'app/contexts/ImageUploaderTriggerContext'; import { activeTabNameSelector } from 'features/options/store/optionsSelectors'; import { tabDict } from 'features/tabs/components/InvokeTabs'; import ImageUploadOverlay from './ImageUploadOverlay'; import { uploadImage } from 'features/gallery/store/thunks/uploadImage'; import useImageUploader from 'common/hooks/useImageUploader'; type ImageUploaderProps = { children: ReactNode; }; const ImageUploader = (props: ImageUploaderProps) => { const { children } = props; const dispatch = useAppDispatch(); const activeTabName = useAppSelector(activeTabNameSelector); const toast = useToast({}); const [isHandlingUpload, setIsHandlingUpload] = useState(false); const { setOpenUploader } = useImageUploader(); const fileRejectionCallback = useCallback( (rejection: FileRejection) => { setIsHandlingUpload(true); const msg = rejection.errors.reduce( (acc: string, cur: { message: string }) => acc + '\n' + cur.message, '' ); toast({ title: 'Upload failed', description: msg, status: 'error', isClosable: true, }); }, [toast] ); const fileAcceptedCallback = useCallback( async (file: File) => { dispatch(uploadImage({ imageFile: file })); }, [dispatch] ); const onDrop = useCallback( (acceptedFiles: Array, fileRejections: Array) => { fileRejections.forEach((rejection: FileRejection) => { fileRejectionCallback(rejection); }); acceptedFiles.forEach((file: File) => { fileAcceptedCallback(file); }); }, [fileAcceptedCallback, fileRejectionCallback] ); const { getRootProps, getInputProps, isDragAccept, isDragReject, isDragActive, open, } = useDropzone({ accept: { 'image/png': ['.png'], 'image/jpeg': ['.jpg', '.jpeg', '.png'] }, noClick: true, onDrop, onDragOver: () => setIsHandlingUpload(true), maxFiles: 1, }); setOpenUploader(open); useEffect(() => { const pasteImageListener = (e: ClipboardEvent) => { const dataTransferItemList = e.clipboardData?.items; if (!dataTransferItemList) return; const imageItems: Array = []; for (const item of dataTransferItemList) { if ( item.kind === 'file' && ['image/png', 'image/jpg'].includes(item.type) ) { imageItems.push(item); } } if (!imageItems.length) return; e.stopImmediatePropagation(); if (imageItems.length > 1) { toast({ description: 'Multiple images pasted, may only upload one image at a time', status: 'error', isClosable: true, }); return; } const file = imageItems[0].getAsFile(); if (!file) { toast({ description: 'Unable to load file', status: 'error', isClosable: true, }); return; } dispatch(uploadImage({ imageFile: file })); }; document.addEventListener('paste', pasteImageListener); return () => { document.removeEventListener('paste', pasteImageListener); }; }, [dispatch, toast, activeTabName]); const overlaySecondaryText = ['img2img', 'unifiedCanvas'].includes( activeTabName ) ? ` to ${tabDict[activeTabName as keyof typeof tabDict].tooltip}` : ``; return (
{ // Bail out if user hits spacebar - do not open the uploader if (e.key === ' ') return; }} > {children} {isDragActive && isHandlingUpload && ( )}
); }; export default ImageUploader;