mirror of
https://github.com/invoke-ai/InvokeAI.git
synced 2026-02-06 08:35:10 -05:00
130 lines
4.3 KiB
TypeScript
130 lines
4.3 KiB
TypeScript
import { Box, Flex } from '@invoke-ai/ui-library';
|
|
import { createSelector } from '@reduxjs/toolkit';
|
|
import { skipToken } from '@reduxjs/toolkit/query';
|
|
import { useAppSelector } from 'app/store/storeHooks';
|
|
import IAIDndImage from 'common/components/IAIDndImage';
|
|
import { IAINoContentFallback } from 'common/components/IAIImageFallback';
|
|
import type { TypesafeDraggableData, TypesafeDroppableData } from 'features/dnd/types';
|
|
import ProgressImage from 'features/gallery/components/CurrentImage/ProgressImage';
|
|
import ImageMetadataViewer from 'features/gallery/components/ImageMetadataViewer/ImageMetadataViewer';
|
|
import NextPrevImageButtons from 'features/gallery/components/NextPrevImageButtons';
|
|
import { selectLastSelectedImage } from 'features/gallery/store/gallerySelectors';
|
|
import type { AnimationProps } from 'framer-motion';
|
|
import { AnimatePresence, motion } from 'framer-motion';
|
|
import type { CSSProperties } from 'react';
|
|
import { memo, useCallback, useMemo, useRef, useState } from 'react';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { PiImageBold } from 'react-icons/pi';
|
|
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
|
|
|
const selectLastSelectedImageName = createSelector(
|
|
selectLastSelectedImage,
|
|
(lastSelectedImage) => lastSelectedImage?.image_name
|
|
);
|
|
|
|
const CurrentImagePreview = () => {
|
|
const shouldShowImageDetails = useAppSelector((s) => s.ui.shouldShowImageDetails);
|
|
const imageName = useAppSelector(selectLastSelectedImageName);
|
|
const hasDenoiseProgress = useAppSelector((s) => Boolean(s.system.denoiseProgress));
|
|
const shouldShowProgressInViewer = useAppSelector((s) => s.ui.shouldShowProgressInViewer);
|
|
|
|
const { currentData: imageDTO } = useGetImageDTOQuery(imageName ?? skipToken);
|
|
|
|
const draggableData = useMemo<TypesafeDraggableData | undefined>(() => {
|
|
if (imageDTO) {
|
|
return {
|
|
id: 'current-image',
|
|
payloadType: 'IMAGE_DTO',
|
|
payload: { imageDTO },
|
|
};
|
|
}
|
|
}, [imageDTO]);
|
|
|
|
const droppableData = useMemo<TypesafeDroppableData | undefined>(
|
|
() => ({
|
|
id: 'current-image',
|
|
actionType: 'SET_CURRENT_IMAGE',
|
|
}),
|
|
[]
|
|
);
|
|
|
|
// Show and hide the next/prev buttons on mouse move
|
|
const [shouldShowNextPrevButtons, setShouldShowNextPrevButtons] = useState<boolean>(false);
|
|
|
|
const timeoutId = useRef(0);
|
|
|
|
const { t } = useTranslation();
|
|
|
|
const handleMouseOver = useCallback(() => {
|
|
setShouldShowNextPrevButtons(true);
|
|
window.clearTimeout(timeoutId.current);
|
|
}, []);
|
|
|
|
const handleMouseOut = useCallback(() => {
|
|
timeoutId.current = window.setTimeout(() => {
|
|
setShouldShowNextPrevButtons(false);
|
|
}, 500);
|
|
}, []);
|
|
|
|
return (
|
|
<Flex
|
|
onMouseOver={handleMouseOver}
|
|
onMouseOut={handleMouseOut}
|
|
width="full"
|
|
height="full"
|
|
alignItems="center"
|
|
justifyContent="center"
|
|
position="relative"
|
|
>
|
|
{hasDenoiseProgress && shouldShowProgressInViewer ? (
|
|
<ProgressImage />
|
|
) : (
|
|
<IAIDndImage
|
|
imageDTO={imageDTO}
|
|
droppableData={droppableData}
|
|
draggableData={draggableData}
|
|
isUploadDisabled={true}
|
|
fitContainer
|
|
useThumbailFallback
|
|
dropLabel={t('gallery.setCurrentImage')}
|
|
noContentFallback={<IAINoContentFallback icon={PiImageBold} label={t('gallery.noImageSelected')} />}
|
|
dataTestId="image-preview"
|
|
/>
|
|
)}
|
|
{shouldShowImageDetails && imageDTO && (
|
|
<Box position="absolute" top="0" width="full" height="full" borderRadius="base">
|
|
<ImageMetadataViewer image={imageDTO} />
|
|
</Box>
|
|
)}
|
|
<AnimatePresence>
|
|
{!shouldShowImageDetails && imageDTO && shouldShowNextPrevButtons && (
|
|
<motion.div key="nextPrevButtons" initial={initial} animate={animate} exit={exit} style={motionStyles}>
|
|
<NextPrevImageButtons />
|
|
</motion.div>
|
|
)}
|
|
</AnimatePresence>
|
|
</Flex>
|
|
);
|
|
};
|
|
|
|
export default memo(CurrentImagePreview);
|
|
|
|
const initial: AnimationProps['initial'] = {
|
|
opacity: 0,
|
|
};
|
|
const animate: AnimationProps['animate'] = {
|
|
opacity: 1,
|
|
transition: { duration: 0.1 },
|
|
};
|
|
const exit: AnimationProps['exit'] = {
|
|
opacity: 0,
|
|
transition: { duration: 0.1 },
|
|
};
|
|
const motionStyles: CSSProperties = {
|
|
position: 'absolute',
|
|
top: '0',
|
|
width: '100%',
|
|
height: '100%',
|
|
pointerEvents: 'none',
|
|
};
|