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:
psychedelicious 2023-06-24 15:57:13 +10:00
parent 0472b33164
commit dde497404b
6 changed files with 179 additions and 140 deletions

View File

@ -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={{

View File

@ -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 };
};

View File

@ -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={{

View File

@ -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>
); );
}; };

View File

@ -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

View File

@ -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>
); );
}; };