mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
fix(ui): fix init image display buttons
- Reset and Upload buttons along top of initial image - Also had to mess around with the control net & DnD image stuff after changing the styles - Abstract image upload logic into hook - does not handle native HTML drag and drop upload - only the button click upload
This commit is contained in:
parent
0472b33164
commit
dde497404b
@ -12,15 +12,14 @@ import IAIIconButton from 'common/components/IAIIconButton';
|
|||||||
import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
|
import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
|
||||||
import ImageMetadataOverlay from 'common/components/ImageMetadataOverlay';
|
import ImageMetadataOverlay from 'common/components/ImageMetadataOverlay';
|
||||||
import { AnimatePresence } from 'framer-motion';
|
import { AnimatePresence } from 'framer-motion';
|
||||||
import { ReactElement, SyntheticEvent, useCallback } from 'react';
|
import { ReactElement, SyntheticEvent } from 'react';
|
||||||
import { memo, useRef } from 'react';
|
import { memo, useRef } from 'react';
|
||||||
import { FaImage, FaTimes, FaUndo, FaUpload } from 'react-icons/fa';
|
import { FaImage, FaUndo, FaUpload } from 'react-icons/fa';
|
||||||
import { ImageDTO } from 'services/api/types';
|
import { ImageDTO } from 'services/api/types';
|
||||||
import { v4 as uuidv4 } from 'uuid';
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
import IAIDropOverlay from './IAIDropOverlay';
|
import IAIDropOverlay from './IAIDropOverlay';
|
||||||
import { PostUploadAction, imageUploaded } from 'services/api/thunks/image';
|
import { PostUploadAction } from 'services/api/thunks/image';
|
||||||
import { useDropzone } from 'react-dropzone';
|
import { useImageUploadButton } from 'common/hooks/useImageUploadButton';
|
||||||
import { useAppDispatch } from 'app/store/storeHooks';
|
|
||||||
|
|
||||||
type IAIDndImageProps = {
|
type IAIDndImageProps = {
|
||||||
image: ImageDTO | null | undefined;
|
image: ImageDTO | null | undefined;
|
||||||
@ -39,6 +38,7 @@ type IAIDndImageProps = {
|
|||||||
minSize?: number;
|
minSize?: number;
|
||||||
postUploadAction?: PostUploadAction;
|
postUploadAction?: PostUploadAction;
|
||||||
imageSx?: ChakraProps['sx'];
|
imageSx?: ChakraProps['sx'];
|
||||||
|
fitContainer?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
const IAIDndImage = (props: IAIDndImageProps) => {
|
const IAIDndImage = (props: IAIDndImageProps) => {
|
||||||
@ -58,8 +58,9 @@ const IAIDndImage = (props: IAIDndImageProps) => {
|
|||||||
minSize = 24,
|
minSize = 24,
|
||||||
postUploadAction,
|
postUploadAction,
|
||||||
imageSx,
|
imageSx,
|
||||||
|
fitContainer = false,
|
||||||
} = props;
|
} = props;
|
||||||
const dispatch = useAppDispatch();
|
|
||||||
const dndId = useRef(uuidv4());
|
const dndId = useRef(uuidv4());
|
||||||
|
|
||||||
const {
|
const {
|
||||||
@ -87,31 +88,9 @@ const IAIDndImage = (props: IAIDndImageProps) => {
|
|||||||
disabled: isDragDisabled || !image,
|
disabled: isDragDisabled || !image,
|
||||||
});
|
});
|
||||||
|
|
||||||
const handleOnDropAccepted = useCallback(
|
const { getUploadButtonProps, getUploadInputProps } = useImageUploadButton({
|
||||||
(files: Array<File>) => {
|
|
||||||
const file = files[0];
|
|
||||||
if (!file) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
dispatch(
|
|
||||||
imageUploaded({
|
|
||||||
file,
|
|
||||||
image_category: 'user',
|
|
||||||
is_intermediate: false,
|
|
||||||
postUploadAction,
|
postUploadAction,
|
||||||
})
|
isDisabled: isUploadDisabled,
|
||||||
);
|
|
||||||
},
|
|
||||||
[dispatch, postUploadAction]
|
|
||||||
);
|
|
||||||
|
|
||||||
const { getRootProps, getInputProps } = useDropzone({
|
|
||||||
accept: { 'image/png': ['.png'], 'image/jpeg': ['.jpg', '.jpeg', '.png'] },
|
|
||||||
onDropAccepted: handleOnDropAccepted,
|
|
||||||
noDrag: true,
|
|
||||||
multiple: false,
|
|
||||||
disabled: isUploadDisabled,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const setNodeRef = useCombinedRefs(setDroppableRef, setDraggableRef);
|
const setNodeRef = useCombinedRefs(setDroppableRef, setDraggableRef);
|
||||||
@ -149,13 +128,13 @@ const IAIDndImage = (props: IAIDndImageProps) => {
|
|||||||
sx={{
|
sx={{
|
||||||
w: 'full',
|
w: 'full',
|
||||||
h: 'full',
|
h: 'full',
|
||||||
|
position: fitContainer ? 'absolute' : 'relative',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
justifyContent: 'center',
|
justifyContent: 'center',
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Image
|
<Image
|
||||||
src={image.image_url}
|
src={image.image_url}
|
||||||
fallbackStrategy="beforeLoadOrError"
|
|
||||||
fallback={fallback}
|
fallback={fallback}
|
||||||
onError={onError}
|
onError={onError}
|
||||||
objectFit="contain"
|
objectFit="contain"
|
||||||
@ -205,9 +184,9 @@ const IAIDndImage = (props: IAIDndImageProps) => {
|
|||||||
color: 'base.500',
|
color: 'base.500',
|
||||||
...uploadButtonStyles,
|
...uploadButtonStyles,
|
||||||
}}
|
}}
|
||||||
{...getRootProps()}
|
{...getUploadButtonProps()}
|
||||||
>
|
>
|
||||||
<input {...getInputProps()} />
|
<input {...getUploadInputProps()} />
|
||||||
<Icon
|
<Icon
|
||||||
as={isUploadDisabled ? FaImage : FaUpload}
|
as={isUploadDisabled ? FaImage : FaUpload}
|
||||||
sx={{
|
sx={{
|
||||||
|
@ -0,0 +1,64 @@
|
|||||||
|
import { useAppDispatch } from 'app/store/storeHooks';
|
||||||
|
import { useCallback } from 'react';
|
||||||
|
import { useDropzone } from 'react-dropzone';
|
||||||
|
import { PostUploadAction, imageUploaded } from 'services/api/thunks/image';
|
||||||
|
|
||||||
|
type UseImageUploadButtonArgs = {
|
||||||
|
postUploadAction?: PostUploadAction;
|
||||||
|
isDisabled?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides image uploader functionality to any component.
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* const { getUploadButtonProps, getUploadInputProps } = useImageUploadButton({
|
||||||
|
* postUploadAction: {
|
||||||
|
* type: 'SET_CONTROLNET_IMAGE',
|
||||||
|
* controlNetId: '12345',
|
||||||
|
* },
|
||||||
|
* isDisabled: getIsUploadDisabled(),
|
||||||
|
* });
|
||||||
|
*
|
||||||
|
* // in the render function
|
||||||
|
* <Button {...getUploadButtonProps()} /> // will open the file dialog on click
|
||||||
|
* <input {...getUploadInputProps()} /> // hidden, handles native upload functionality
|
||||||
|
*/
|
||||||
|
export const useImageUploadButton = ({
|
||||||
|
postUploadAction,
|
||||||
|
isDisabled,
|
||||||
|
}: UseImageUploadButtonArgs) => {
|
||||||
|
const dispatch = useAppDispatch();
|
||||||
|
const onDropAccepted = useCallback(
|
||||||
|
(files: File[]) => {
|
||||||
|
const file = files[0];
|
||||||
|
if (!file) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
dispatch(
|
||||||
|
imageUploaded({
|
||||||
|
file,
|
||||||
|
image_category: 'user',
|
||||||
|
is_intermediate: false,
|
||||||
|
postUploadAction,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[dispatch, postUploadAction]
|
||||||
|
);
|
||||||
|
|
||||||
|
const {
|
||||||
|
getRootProps: getUploadButtonProps,
|
||||||
|
getInputProps: getUploadInputProps,
|
||||||
|
open: openUploader,
|
||||||
|
} = useDropzone({
|
||||||
|
accept: { 'image/png': ['.png'], 'image/jpeg': ['.jpg', '.jpeg', '.png'] },
|
||||||
|
onDropAccepted,
|
||||||
|
disabled: isDisabled,
|
||||||
|
noDrag: true,
|
||||||
|
multiple: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
return { getUploadButtonProps, getUploadInputProps, openUploader };
|
||||||
|
};
|
@ -10,7 +10,6 @@ import { Box, ChakraProps, Flex } from '@chakra-ui/react';
|
|||||||
import IAIDndImage from 'common/components/IAIDndImage';
|
import IAIDndImage from 'common/components/IAIDndImage';
|
||||||
import { createSelector } from '@reduxjs/toolkit';
|
import { createSelector } from '@reduxjs/toolkit';
|
||||||
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
|
import { defaultSelectorOptions } from 'app/store/util/defaultMemoizeOptions';
|
||||||
import { AnimatePresence, motion } from 'framer-motion';
|
|
||||||
import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
|
import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
|
||||||
import IAIIconButton from 'common/components/IAIIconButton';
|
import IAIIconButton from 'common/components/IAIIconButton';
|
||||||
import { FaUndo } from 'react-icons/fa';
|
import { FaUndo } from 'react-icons/fa';
|
||||||
@ -105,30 +104,24 @@ const ControlNetImagePreview = (props: Props) => {
|
|||||||
<IAIDndImage
|
<IAIDndImage
|
||||||
image={controlImage}
|
image={controlImage}
|
||||||
onDrop={handleDrop}
|
onDrop={handleDrop}
|
||||||
isDropDisabled={Boolean(
|
isDropDisabled={shouldShowProcessedImage}
|
||||||
processedControlImage && processorType !== 'none'
|
|
||||||
)}
|
|
||||||
isUploadDisabled={Boolean(controlImage)}
|
isUploadDisabled={Boolean(controlImage)}
|
||||||
postUploadAction={{ type: 'SET_CONTROLNET_IMAGE', controlNetId }}
|
postUploadAction={{ type: 'SET_CONTROLNET_IMAGE', controlNetId }}
|
||||||
imageSx={imageSx}
|
imageSx={{
|
||||||
|
pointerEvents: shouldShowProcessedImage ? 'none' : 'auto',
|
||||||
|
...imageSx,
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
<AnimatePresence>
|
|
||||||
{shouldShowProcessedImage && (
|
{shouldShowProcessedImage && (
|
||||||
<motion.div
|
<Box
|
||||||
style={{ width: '100%' }}
|
sx={{
|
||||||
initial={{
|
position: 'absolute',
|
||||||
opacity: 0,
|
top: 0,
|
||||||
}}
|
insetInlineStart: 0,
|
||||||
animate={{
|
w: 'full',
|
||||||
opacity: 1,
|
h: 'full',
|
||||||
transition: { duration: 0.1 },
|
|
||||||
}}
|
|
||||||
exit={{
|
|
||||||
opacity: 0,
|
|
||||||
transition: { duration: 0.1 },
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<>
|
|
||||||
{shouldShowProcessedImageBackdrop && (
|
{shouldShowProcessedImageBackdrop && (
|
||||||
<Box
|
<Box
|
||||||
sx={{
|
sx={{
|
||||||
@ -142,15 +135,6 @@ const ControlNetImagePreview = (props: Props) => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<Box
|
|
||||||
sx={{
|
|
||||||
position: 'absolute',
|
|
||||||
top: 0,
|
|
||||||
insetInlineStart: 0,
|
|
||||||
w: 'full',
|
|
||||||
h: 'full',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<IAIDndImage
|
<IAIDndImage
|
||||||
image={processedControlImage}
|
image={processedControlImage}
|
||||||
onDrop={handleDrop}
|
onDrop={handleDrop}
|
||||||
@ -159,10 +143,7 @@ const ControlNetImagePreview = (props: Props) => {
|
|||||||
imageSx={imageSx}
|
imageSx={imageSx}
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
</>
|
|
||||||
</motion.div>
|
|
||||||
)}
|
)}
|
||||||
</AnimatePresence>
|
|
||||||
{pendingControlImages.includes(controlNetId) && (
|
{pendingControlImages.includes(controlNetId) && (
|
||||||
<Box
|
<Box
|
||||||
sx={{
|
sx={{
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
import { Box, Flex } from '@chakra-ui/react';
|
import { Flex } from '@chakra-ui/react';
|
||||||
import { createSelector } from '@reduxjs/toolkit';
|
import { createSelector } from '@reduxjs/toolkit';
|
||||||
import { useAppSelector } from 'app/store/storeHooks';
|
import { useAppSelector } from 'app/store/storeHooks';
|
||||||
import { systemSelector } from 'features/system/store/systemSelectors';
|
import { systemSelector } from 'features/system/store/systemSelectors';
|
||||||
import { isEqual } from 'lodash-es';
|
|
||||||
|
|
||||||
import { gallerySelector } from '../store/gallerySelectors';
|
import { gallerySelector } from '../store/gallerySelectors';
|
||||||
import CurrentImageButtons from './CurrentImageButtons';
|
import CurrentImageButtons from './CurrentImageButtons';
|
||||||
@ -22,13 +21,8 @@ export const currentImageDisplaySelector = createSelector(
|
|||||||
defaultSelectorOptions
|
defaultSelectorOptions
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
|
||||||
* Displays the current image if there is one, plus associated actions.
|
|
||||||
*/
|
|
||||||
const CurrentImageDisplay = () => {
|
const CurrentImageDisplay = () => {
|
||||||
const { hasSelectedImage, hasProgressImage } = useAppSelector(
|
const { hasSelectedImage } = useAppSelector(currentImageDisplaySelector);
|
||||||
currentImageDisplaySelector
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex
|
<Flex
|
||||||
@ -43,25 +37,9 @@ const CurrentImageDisplay = () => {
|
|||||||
justifyContent: 'center',
|
justifyContent: 'center',
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Flex
|
{hasSelectedImage && <CurrentImageButtons />}
|
||||||
flexDirection="column"
|
|
||||||
sx={{
|
|
||||||
w: 'full',
|
|
||||||
h: 'full',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'center',
|
|
||||||
gap: 4,
|
|
||||||
position: 'absolute',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<CurrentImagePreview />
|
<CurrentImagePreview />
|
||||||
</Flex>
|
</Flex>
|
||||||
{hasSelectedImage && (
|
|
||||||
<Box sx={{ position: 'absolute', top: 0 }}>
|
|
||||||
<CurrentImageButtons />
|
|
||||||
</Box>
|
|
||||||
)}
|
|
||||||
</Flex>
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -51,10 +51,6 @@ const CurrentImagePreview = () => {
|
|||||||
shouldAntialiasProgressImage,
|
shouldAntialiasProgressImage,
|
||||||
} = useAppSelector(imagesSelector);
|
} = useAppSelector(imagesSelector);
|
||||||
|
|
||||||
// const image = useAppSelector((state: RootState) =>
|
|
||||||
// selectImagesById(state, selectedImage ?? '')
|
|
||||||
// );
|
|
||||||
|
|
||||||
const {
|
const {
|
||||||
currentData: image,
|
currentData: image,
|
||||||
isLoading,
|
isLoading,
|
||||||
@ -79,7 +75,6 @@ const CurrentImagePreview = () => {
|
|||||||
sx={{
|
sx={{
|
||||||
width: 'full',
|
width: 'full',
|
||||||
height: 'full',
|
height: 'full',
|
||||||
position: 'relative',
|
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
justifyContent: 'center',
|
justifyContent: 'center',
|
||||||
}}
|
}}
|
||||||
@ -101,21 +96,13 @@ const CurrentImagePreview = () => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<Flex
|
|
||||||
sx={{
|
|
||||||
width: 'full',
|
|
||||||
height: 'full',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'center',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<IAIDndImage
|
<IAIDndImage
|
||||||
image={image}
|
image={image}
|
||||||
onDrop={handleDrop}
|
onDrop={handleDrop}
|
||||||
fallback={<IAIImageLoadingFallback sx={{ bg: 'none' }} />}
|
fallback={<IAIImageLoadingFallback sx={{ bg: 'none' }} />}
|
||||||
isUploadDisabled={true}
|
isUploadDisabled={true}
|
||||||
|
fitContainer
|
||||||
/>
|
/>
|
||||||
</Flex>
|
|
||||||
)}
|
)}
|
||||||
{shouldShowImageDetails && image && (
|
{shouldShowImageDetails && image && (
|
||||||
<Box
|
<Box
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { Flex } from '@chakra-ui/react';
|
import { Flex, Spacer, Text } from '@chakra-ui/react';
|
||||||
import { createSelector } from '@reduxjs/toolkit';
|
import { createSelector } from '@reduxjs/toolkit';
|
||||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||||
import {
|
import {
|
||||||
@ -13,6 +13,10 @@ import { ImageDTO } from 'services/api/types';
|
|||||||
import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
|
import { IAIImageLoadingFallback } from 'common/components/IAIImageFallback';
|
||||||
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
||||||
import { skipToken } from '@reduxjs/toolkit/dist/query';
|
import { skipToken } from '@reduxjs/toolkit/dist/query';
|
||||||
|
import IAIIconButton from 'common/components/IAIIconButton';
|
||||||
|
import { FaUndo, FaUpload } from 'react-icons/fa';
|
||||||
|
import useImageUploader from 'common/hooks/useImageUploader';
|
||||||
|
import { useImageUploadButton } from 'common/hooks/useImageUploadButton';
|
||||||
|
|
||||||
const selector = createSelector(
|
const selector = createSelector(
|
||||||
[generationSelector],
|
[generationSelector],
|
||||||
@ -28,6 +32,7 @@ const selector = createSelector(
|
|||||||
const InitialImagePreview = () => {
|
const InitialImagePreview = () => {
|
||||||
const { initialImage } = useAppSelector(selector);
|
const { initialImage } = useAppSelector(selector);
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
|
const { openUploader } = useImageUploader();
|
||||||
|
|
||||||
const {
|
const {
|
||||||
currentData: image,
|
currentData: image,
|
||||||
@ -36,6 +41,10 @@ const InitialImagePreview = () => {
|
|||||||
isSuccess,
|
isSuccess,
|
||||||
} = useGetImageDTOQuery(initialImage?.imageName ?? skipToken);
|
} = useGetImageDTOQuery(initialImage?.imageName ?? skipToken);
|
||||||
|
|
||||||
|
const { getUploadButtonProps, getUploadInputProps } = useImageUploadButton({
|
||||||
|
postUploadAction: { type: 'SET_INITIAL_IMAGE' },
|
||||||
|
});
|
||||||
|
|
||||||
const handleDrop = useCallback(
|
const handleDrop = useCallback(
|
||||||
(droppedImage: ImageDTO) => {
|
(droppedImage: ImageDTO) => {
|
||||||
if (droppedImage.image_name === initialImage?.imageName) {
|
if (droppedImage.image_name === initialImage?.imageName) {
|
||||||
@ -50,25 +59,66 @@ const InitialImagePreview = () => {
|
|||||||
dispatch(clearInitialImage());
|
dispatch(clearInitialImage());
|
||||||
}, [dispatch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
|
const handleUpload = useCallback(() => {
|
||||||
|
openUploader();
|
||||||
|
}, [openUploader]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex
|
<Flex
|
||||||
sx={{
|
sx={{
|
||||||
|
flexDir: 'column',
|
||||||
width: 'full',
|
width: 'full',
|
||||||
height: 'full',
|
height: 'full',
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
alignItems: 'center',
|
alignItems: 'center',
|
||||||
justifyContent: 'center',
|
justifyContent: 'center',
|
||||||
p: 4,
|
p: 4,
|
||||||
|
gap: 4,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
<Flex
|
||||||
|
sx={{
|
||||||
|
w: 'full',
|
||||||
|
flexWrap: 'wrap',
|
||||||
|
justifyContent: 'center',
|
||||||
|
alignItems: 'center',
|
||||||
|
gap: 2,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
sx={{
|
||||||
|
color: 'base.200',
|
||||||
|
fontWeight: 600,
|
||||||
|
fontSize: 'sm',
|
||||||
|
userSelect: 'none',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Initial Image
|
||||||
|
</Text>
|
||||||
|
<Spacer />
|
||||||
|
<IAIIconButton
|
||||||
|
tooltip="Upload Initial Image"
|
||||||
|
aria-label="Upload Initial Image"
|
||||||
|
icon={<FaUpload />}
|
||||||
|
onClick={handleUpload}
|
||||||
|
{...getUploadButtonProps()}
|
||||||
|
/>
|
||||||
|
<IAIIconButton
|
||||||
|
tooltip="Reset Initial Image"
|
||||||
|
aria-label="Reset Initial Image"
|
||||||
|
icon={<FaUndo />}
|
||||||
|
onClick={handleReset}
|
||||||
|
isDisabled={!initialImage}
|
||||||
|
/>
|
||||||
|
</Flex>
|
||||||
<IAIDndImage
|
<IAIDndImage
|
||||||
image={image}
|
image={image}
|
||||||
onDrop={handleDrop}
|
onDrop={handleDrop}
|
||||||
onReset={handleReset}
|
|
||||||
fallback={<IAIImageLoadingFallback sx={{ bg: 'none' }} />}
|
fallback={<IAIImageLoadingFallback sx={{ bg: 'none' }} />}
|
||||||
postUploadAction={{ type: 'SET_INITIAL_IMAGE' }}
|
isUploadDisabled={true}
|
||||||
withResetIcon
|
fitContainer
|
||||||
/>
|
/>
|
||||||
|
<input {...getUploadInputProps()} />
|
||||||
</Flex>
|
</Flex>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user