mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
feat(ui): rework visibility conditions for image viewer
This commit is contained in:
parent
c325ad3432
commit
038a482ef0
@ -380,7 +380,7 @@
|
|||||||
"problemDeletingImagesDesc": "One or more images could not be deleted",
|
"problemDeletingImagesDesc": "One or more images could not be deleted",
|
||||||
"viewerImage": "Viewer Image",
|
"viewerImage": "Viewer Image",
|
||||||
"compareImage": "Compare Image",
|
"compareImage": "Compare Image",
|
||||||
"selectForViewer": "Select for Viewer",
|
"openInViewer": "Open in Viewer",
|
||||||
"selectForCompare": "Select for Compare",
|
"selectForCompare": "Select for Compare",
|
||||||
"selectAnImageToCompare": "Select an Image to Compare",
|
"selectAnImageToCompare": "Select an Image to Compare",
|
||||||
"slider": "Slider",
|
"slider": "Slider",
|
||||||
|
@ -7,6 +7,7 @@ import {
|
|||||||
imageToCompareChanged,
|
imageToCompareChanged,
|
||||||
} from 'features/gallery/store/gallerySlice';
|
} from 'features/gallery/store/gallerySlice';
|
||||||
import { memo, useCallback } from 'react';
|
import { memo, useCallback } from 'react';
|
||||||
|
import { useHotkeys } from 'react-hotkeys-hook';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { PiArrowsOutBold, PiSwapBold, PiXBold } from 'react-icons/pi';
|
import { PiArrowsOutBold, PiSwapBold, PiXBold } from 'react-icons/pi';
|
||||||
|
|
||||||
@ -33,6 +34,7 @@ export const CompareToolbar = memo(() => {
|
|||||||
const exitCompare = useCallback(() => {
|
const exitCompare = useCallback(() => {
|
||||||
dispatch(imageToCompareChanged(null));
|
dispatch(imageToCompareChanged(null));
|
||||||
}, [dispatch]);
|
}, [dispatch]);
|
||||||
|
useHotkeys('esc', exitCompare, [exitCompare]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex w="full" gap={2}>
|
<Flex w="full" gap={2}>
|
||||||
|
@ -1,21 +1,14 @@
|
|||||||
import { createMemoizedSelector } from 'app/store/createMemoizedSelector';
|
|
||||||
import { useAppSelector } from 'app/store/storeHooks';
|
import { useAppSelector } from 'app/store/storeHooks';
|
||||||
import { IAINoContentFallback } from 'common/components/IAIImageFallback';
|
import { IAINoContentFallback } from 'common/components/IAIImageFallback';
|
||||||
import type { Dimensions } from 'features/canvas/store/canvasTypes';
|
import type { Dimensions } from 'features/canvas/store/canvasTypes';
|
||||||
|
import { selectComparisonImages } from 'features/gallery/components/ImageViewer/common';
|
||||||
import { ImageComparisonHover } from 'features/gallery/components/ImageViewer/ImageComparisonHover';
|
import { ImageComparisonHover } from 'features/gallery/components/ImageViewer/ImageComparisonHover';
|
||||||
import { ImageComparisonSideBySide } from 'features/gallery/components/ImageViewer/ImageComparisonSideBySide';
|
import { ImageComparisonSideBySide } from 'features/gallery/components/ImageViewer/ImageComparisonSideBySide';
|
||||||
import { ImageComparisonSlider } from 'features/gallery/components/ImageViewer/ImageComparisonSlider';
|
import { ImageComparisonSlider } from 'features/gallery/components/ImageViewer/ImageComparisonSlider';
|
||||||
import { selectGallerySlice } from 'features/gallery/store/gallerySlice';
|
|
||||||
import { memo } from 'react';
|
import { memo } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { PiImagesBold } from 'react-icons/pi';
|
import { PiImagesBold } from 'react-icons/pi';
|
||||||
|
|
||||||
const selector = createMemoizedSelector(selectGallerySlice, (gallerySlice) => {
|
|
||||||
const firstImage = gallerySlice.selection.slice(-1)[0] ?? null;
|
|
||||||
const secondImage = gallerySlice.imageToCompare;
|
|
||||||
return { firstImage, secondImage };
|
|
||||||
});
|
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
containerDims: Dimensions;
|
containerDims: Dimensions;
|
||||||
};
|
};
|
||||||
@ -23,7 +16,7 @@ type Props = {
|
|||||||
export const ImageComparison = memo(({ containerDims }: Props) => {
|
export const ImageComparison = memo(({ containerDims }: Props) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const comparisonMode = useAppSelector((s) => s.gallery.comparisonMode);
|
const comparisonMode = useAppSelector((s) => s.gallery.comparisonMode);
|
||||||
const { firstImage, secondImage } = useAppSelector(selector);
|
const { firstImage, secondImage } = useAppSelector(selectComparisonImages);
|
||||||
|
|
||||||
if (!firstImage || !secondImage) {
|
if (!firstImage || !secondImage) {
|
||||||
// Should rarely/never happen - we don't render this component unless we have images to compare
|
// Should rarely/never happen - we don't render this component unless we have images to compare
|
||||||
|
@ -1,17 +1,12 @@
|
|||||||
import { Flex } from '@invoke-ai/ui-library';
|
import { Flex } from '@invoke-ai/ui-library';
|
||||||
import { createMemoizedSelector } from 'app/store/createMemoizedSelector';
|
|
||||||
import { useAppSelector } from 'app/store/storeHooks';
|
import { useAppSelector } from 'app/store/storeHooks';
|
||||||
import IAIDroppable from 'common/components/IAIDroppable';
|
import IAIDroppable from 'common/components/IAIDroppable';
|
||||||
import type { CurrentImageDropData, SelectForCompareDropData } from 'features/dnd/types';
|
import type { CurrentImageDropData, SelectForCompareDropData } from 'features/dnd/types';
|
||||||
import { selectGallerySlice } from 'features/gallery/store/gallerySlice';
|
import { useImageViewer } from 'features/gallery/components/ImageViewer/useImageViewer';
|
||||||
import { memo, useMemo } from 'react';
|
import { memo, useMemo } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
|
||||||
const selector = createMemoizedSelector(selectGallerySlice, (gallerySlice) => {
|
import { selectComparisonImages } from './common';
|
||||||
const firstImage = gallerySlice.selection.slice(-1)[0] ?? null;
|
|
||||||
const secondImage = gallerySlice.imageToCompare;
|
|
||||||
return { firstImage, secondImage };
|
|
||||||
});
|
|
||||||
|
|
||||||
const setCurrentImageDropData: CurrentImageDropData = {
|
const setCurrentImageDropData: CurrentImageDropData = {
|
||||||
id: 'current-image',
|
id: 'current-image',
|
||||||
@ -20,7 +15,8 @@ const setCurrentImageDropData: CurrentImageDropData = {
|
|||||||
|
|
||||||
export const ImageComparisonDroppable = memo(() => {
|
export const ImageComparisonDroppable = memo(() => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { firstImage, secondImage } = useAppSelector(selector);
|
const imageViewer = useImageViewer();
|
||||||
|
const { firstImage, secondImage } = useAppSelector(selectComparisonImages);
|
||||||
const selectForCompareDropData = useMemo<SelectForCompareDropData>(
|
const selectForCompareDropData = useMemo<SelectForCompareDropData>(
|
||||||
() => ({
|
() => ({
|
||||||
id: 'image-comparison',
|
id: 'image-comparison',
|
||||||
@ -33,14 +29,17 @@ export const ImageComparisonDroppable = memo(() => {
|
|||||||
[firstImage?.image_name, secondImage?.image_name]
|
[firstImage?.image_name, secondImage?.image_name]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (!imageViewer.isOpen) {
|
||||||
|
return (
|
||||||
|
<Flex position="absolute" top={0} right={0} bottom={0} left={0} gap={2} pointerEvents="none">
|
||||||
|
<IAIDroppable data={setCurrentImageDropData} dropLabel={t('gallery.openInViewer')} />
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex position="absolute" top={0} right={0} bottom={0} left={0} gap={2} pointerEvents="none">
|
<Flex position="absolute" top={0} right={0} bottom={0} left={0} gap={2} pointerEvents="none">
|
||||||
<Flex position="relative" flex={1}>
|
<IAIDroppable data={selectForCompareDropData} dropLabel={t('gallery.selectForCompare')} />
|
||||||
<IAIDroppable data={setCurrentImageDropData} dropLabel={t('gallery.selectForViewer')} />
|
|
||||||
</Flex>
|
|
||||||
<Flex position="relative" flex={1}>
|
|
||||||
<IAIDroppable data={selectForCompareDropData} dropLabel={t('gallery.selectForCompare')} />
|
|
||||||
</Flex>
|
|
||||||
</Flex>
|
</Flex>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
@ -1,43 +1,17 @@
|
|||||||
import { Box, Flex } from '@invoke-ai/ui-library';
|
import { Box, Flex } from '@invoke-ai/ui-library';
|
||||||
import { useAppSelector } from 'app/store/storeHooks';
|
|
||||||
import { CompareToolbar } from 'features/gallery/components/ImageViewer/CompareToolbar';
|
import { CompareToolbar } from 'features/gallery/components/ImageViewer/CompareToolbar';
|
||||||
import CurrentImagePreview from 'features/gallery/components/ImageViewer/CurrentImagePreview';
|
import CurrentImagePreview from 'features/gallery/components/ImageViewer/CurrentImagePreview';
|
||||||
import { ImageComparison } from 'features/gallery/components/ImageViewer/ImageComparison';
|
import { ImageComparison } from 'features/gallery/components/ImageViewer/ImageComparison';
|
||||||
import { ViewerToolbar } from 'features/gallery/components/ImageViewer/ViewerToolbar';
|
import { ViewerToolbar } from 'features/gallery/components/ImageViewer/ViewerToolbar';
|
||||||
import type { InvokeTabName } from 'features/ui/store/tabMap';
|
import { memo } from 'react';
|
||||||
import { activeTabNameSelector } from 'features/ui/store/uiSelectors';
|
|
||||||
import { memo, useMemo } from 'react';
|
|
||||||
import { useHotkeys } from 'react-hotkeys-hook';
|
|
||||||
import { useMeasure } from 'react-use';
|
import { useMeasure } from 'react-use';
|
||||||
|
|
||||||
import { useImageViewer } from './useImageViewer';
|
import { useImageViewer } from './useImageViewer';
|
||||||
|
|
||||||
const VIEWER_ENABLED_TABS: InvokeTabName[] = ['canvas', 'generation', 'workflows'];
|
|
||||||
|
|
||||||
export const ImageViewer = memo(() => {
|
export const ImageViewer = memo(() => {
|
||||||
const { isOpen, onToggle, onClose } = useImageViewer();
|
const imageViewer = useImageViewer();
|
||||||
const activeTabName = useAppSelector(activeTabNameSelector);
|
|
||||||
const workflowsMode = useAppSelector((s) => s.workflow.mode);
|
|
||||||
const isComparing = useAppSelector((s) => s.gallery.imageToCompare !== null);
|
|
||||||
const isViewerEnabled = useMemo(() => VIEWER_ENABLED_TABS.includes(activeTabName), [activeTabName]);
|
|
||||||
const shouldShowViewer = useMemo(() => {
|
|
||||||
if (activeTabName === 'workflows' && workflowsMode === 'view') {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (!isViewerEnabled) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return isOpen;
|
|
||||||
}, [isOpen, isViewerEnabled, workflowsMode, activeTabName]);
|
|
||||||
const [containerRef, containerDims] = useMeasure<HTMLDivElement>();
|
const [containerRef, containerDims] = useMeasure<HTMLDivElement>();
|
||||||
|
|
||||||
useHotkeys('z', onToggle, { enabled: isViewerEnabled }, [isViewerEnabled, onToggle]);
|
|
||||||
useHotkeys('esc', onClose, { enabled: isViewerEnabled }, [isViewerEnabled, onClose]);
|
|
||||||
|
|
||||||
if (!shouldShowViewer) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex
|
<Flex
|
||||||
layerStyle="first"
|
layerStyle="first"
|
||||||
@ -53,11 +27,11 @@ export const ImageViewer = memo(() => {
|
|||||||
alignItems="center"
|
alignItems="center"
|
||||||
justifyContent="center"
|
justifyContent="center"
|
||||||
>
|
>
|
||||||
{isComparing && <CompareToolbar />}
|
{imageViewer.isComparing && <CompareToolbar />}
|
||||||
{!isComparing && <ViewerToolbar />}
|
{!imageViewer.isComparing && <ViewerToolbar />}
|
||||||
<Box ref={containerRef} w="full" h="full">
|
<Box ref={containerRef} w="full" h="full">
|
||||||
{!isComparing && <CurrentImagePreview />}
|
{!imageViewer.isComparing && <CurrentImagePreview />}
|
||||||
{isComparing && <ImageComparison containerDims={containerDims} />}
|
{imageViewer.isComparing && <ImageComparison containerDims={containerDims} />}
|
||||||
</Box>
|
</Box>
|
||||||
</Flex>
|
</Flex>
|
||||||
);
|
);
|
||||||
|
@ -9,33 +9,35 @@ import {
|
|||||||
PopoverTrigger,
|
PopoverTrigger,
|
||||||
Text,
|
Text,
|
||||||
} from '@invoke-ai/ui-library';
|
} from '@invoke-ai/ui-library';
|
||||||
|
import { useImageViewer } from 'features/gallery/components/ImageViewer/useImageViewer';
|
||||||
|
import { useHotkeys } from 'react-hotkeys-hook';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import { PiCaretDownBold, PiCheckBold, PiEyeBold, PiPencilBold } from 'react-icons/pi';
|
import { PiCaretDownBold, PiCheckBold, PiEyeBold, PiPencilBold } from 'react-icons/pi';
|
||||||
|
|
||||||
import { useImageViewer } from './useImageViewer';
|
|
||||||
|
|
||||||
export const ViewerToggleMenu = () => {
|
export const ViewerToggleMenu = () => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { isOpen, onClose, onOpen } = useImageViewer();
|
const imageViewer = useImageViewer();
|
||||||
|
useHotkeys('z', imageViewer.onToggle, [imageViewer]);
|
||||||
|
useHotkeys('esc', imageViewer.onClose, [imageViewer]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Popover isLazy>
|
<Popover isLazy>
|
||||||
<PopoverTrigger>
|
<PopoverTrigger>
|
||||||
<Button variant="outline" data-testid="toggle-viewer-menu-button">
|
<Button variant="outline" data-testid="toggle-viewer-menu-button" pointerEvents="auto">
|
||||||
<Flex gap={3} w="full" alignItems="center">
|
<Flex gap={3} w="full" alignItems="center">
|
||||||
{isOpen ? <Icon as={PiEyeBold} /> : <Icon as={PiPencilBold} />}
|
{imageViewer.isOpen ? <Icon as={PiEyeBold} /> : <Icon as={PiPencilBold} />}
|
||||||
<Text fontSize="md">{isOpen ? t('common.viewing') : t('common.editing')}</Text>
|
<Text fontSize="md">{imageViewer.isOpen ? t('common.viewing') : t('common.editing')}</Text>
|
||||||
<Icon as={PiCaretDownBold} />
|
<Icon as={PiCaretDownBold} />
|
||||||
</Flex>
|
</Flex>
|
||||||
</Button>
|
</Button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent p={2}>
|
<PopoverContent p={2} pointerEvents="auto">
|
||||||
<PopoverArrow />
|
<PopoverArrow />
|
||||||
<PopoverBody>
|
<PopoverBody>
|
||||||
<Flex flexDir="column">
|
<Flex flexDir="column">
|
||||||
<Button onClick={onOpen} variant="ghost" h="auto" w="auto" p={2}>
|
<Button onClick={imageViewer.onOpen} variant="ghost" h="auto" w="auto" p={2}>
|
||||||
<Flex gap={2} w="full">
|
<Flex gap={2} w="full">
|
||||||
<Icon as={PiCheckBold} visibility={isOpen ? 'visible' : 'hidden'} />
|
<Icon as={PiCheckBold} visibility={imageViewer.isOpen ? 'visible' : 'hidden'} />
|
||||||
<Flex flexDir="column" gap={2} alignItems="flex-start">
|
<Flex flexDir="column" gap={2} alignItems="flex-start">
|
||||||
<Text fontWeight="semibold" color="base.100">
|
<Text fontWeight="semibold" color="base.100">
|
||||||
{t('common.viewing')}
|
{t('common.viewing')}
|
||||||
@ -46,9 +48,9 @@ export const ViewerToggleMenu = () => {
|
|||||||
</Flex>
|
</Flex>
|
||||||
</Flex>
|
</Flex>
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={onClose} variant="ghost" h="auto" w="auto" p={2}>
|
<Button onClick={imageViewer.onClose} variant="ghost" h="auto" w="auto" p={2}>
|
||||||
<Flex gap={2} w="full">
|
<Flex gap={2} w="full">
|
||||||
<Icon as={PiCheckBold} visibility={isOpen ? 'hidden' : 'visible'} />
|
<Icon as={PiCheckBold} visibility={imageViewer.isOpen ? 'hidden' : 'visible'} />
|
||||||
<Flex flexDir="column" gap={2} alignItems="flex-start">
|
<Flex flexDir="column" gap={2} alignItems="flex-start">
|
||||||
<Text fontWeight="semibold" color="base.100">
|
<Text fontWeight="semibold" color="base.100">
|
||||||
{t('common.editing')}
|
{t('common.editing')}
|
||||||
|
@ -3,21 +3,13 @@ import { useAppSelector } from 'app/store/storeHooks';
|
|||||||
import { ToggleMetadataViewerButton } from 'features/gallery/components/ImageViewer/ToggleMetadataViewerButton';
|
import { ToggleMetadataViewerButton } from 'features/gallery/components/ImageViewer/ToggleMetadataViewerButton';
|
||||||
import { ToggleProgressButton } from 'features/gallery/components/ImageViewer/ToggleProgressButton';
|
import { ToggleProgressButton } from 'features/gallery/components/ImageViewer/ToggleProgressButton';
|
||||||
import { activeTabNameSelector } from 'features/ui/store/uiSelectors';
|
import { activeTabNameSelector } from 'features/ui/store/uiSelectors';
|
||||||
import { memo, useMemo } from 'react';
|
import { memo } from 'react';
|
||||||
|
|
||||||
import CurrentImageButtons from './CurrentImageButtons';
|
import CurrentImageButtons from './CurrentImageButtons';
|
||||||
import { ViewerToggleMenu } from './ViewerToggleMenu';
|
import { ViewerToggleMenu } from './ViewerToggleMenu';
|
||||||
|
|
||||||
export const ViewerToolbar = memo(() => {
|
export const ViewerToolbar = memo(() => {
|
||||||
const workflowsMode = useAppSelector((s) => s.workflow.mode);
|
const tab = useAppSelector(activeTabNameSelector);
|
||||||
const activeTabName = useAppSelector(activeTabNameSelector);
|
|
||||||
const shouldShowToggleMenu = useMemo(() => {
|
|
||||||
if (activeTabName !== 'workflows') {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return workflowsMode === 'edit';
|
|
||||||
}, [workflowsMode, activeTabName]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex w="full" gap={2}>
|
<Flex w="full" gap={2}>
|
||||||
<Flex flex={1} justifyContent="center">
|
<Flex flex={1} justifyContent="center">
|
||||||
@ -31,7 +23,7 @@ export const ViewerToolbar = memo(() => {
|
|||||||
</Flex>
|
</Flex>
|
||||||
<Flex flex={1} justifyContent="center">
|
<Flex flex={1} justifyContent="center">
|
||||||
<Flex gap={2} marginInlineStart="auto">
|
<Flex gap={2} marginInlineStart="auto">
|
||||||
{shouldShowToggleMenu && <ViewerToggleMenu />}
|
{tab !== 'workflows' && <ViewerToggleMenu />}
|
||||||
</Flex>
|
</Flex>
|
||||||
</Flex>
|
</Flex>
|
||||||
</Flex>
|
</Flex>
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
|
import { createMemoizedSelector } from 'app/store/createMemoizedSelector';
|
||||||
import type { Dimensions } from 'features/canvas/store/canvasTypes';
|
import type { Dimensions } from 'features/canvas/store/canvasTypes';
|
||||||
|
import { selectGallerySlice } from 'features/gallery/store/gallerySlice';
|
||||||
import type { ComparisonFit } from 'features/gallery/store/types';
|
import type { ComparisonFit } from 'features/gallery/store/types';
|
||||||
import type { ImageDTO } from 'services/api/types';
|
import type { ImageDTO } from 'services/api/types';
|
||||||
|
|
||||||
@ -55,3 +57,8 @@ export const getSecondImageDims = (
|
|||||||
|
|
||||||
return { width, height };
|
return { width, height };
|
||||||
};
|
};
|
||||||
|
export const selectComparisonImages = createMemoizedSelector(selectGallerySlice, (gallerySlice) => {
|
||||||
|
const firstImage = gallerySlice.selection.slice(-1)[0] ?? null;
|
||||||
|
const secondImage = gallerySlice.imageToCompare;
|
||||||
|
return { firstImage, secondImage };
|
||||||
|
});
|
||||||
|
@ -27,5 +27,5 @@ export const useImageViewer = () => {
|
|||||||
}
|
}
|
||||||
}, [dispatch, isComparing, isOpen]);
|
}, [dispatch, isComparing, isOpen]);
|
||||||
|
|
||||||
return { isOpen, onOpen, onClose, onToggle };
|
return { isOpen, onOpen, onClose, onToggle, isComparing };
|
||||||
};
|
};
|
||||||
|
@ -1,9 +1,7 @@
|
|||||||
import 'reactflow/dist/style.css';
|
import 'reactflow/dist/style.css';
|
||||||
|
|
||||||
import { Flex } from '@invoke-ai/ui-library';
|
import { Flex } from '@invoke-ai/ui-library';
|
||||||
import { createMemoizedSelector } from 'app/store/createMemoizedSelector';
|
|
||||||
import { useAppSelector } from 'app/store/storeHooks';
|
import { useAppSelector } from 'app/store/storeHooks';
|
||||||
import { selectWorkflowSlice } from 'features/nodes/store/workflowSlice';
|
|
||||||
import QueueControls from 'features/queue/components/QueueControls';
|
import QueueControls from 'features/queue/components/QueueControls';
|
||||||
import ResizeHandle from 'features/ui/components/tabs/ResizeHandle';
|
import ResizeHandle from 'features/ui/components/tabs/ResizeHandle';
|
||||||
import { usePanelStorage } from 'features/ui/hooks/usePanelStorage';
|
import { usePanelStorage } from 'features/ui/hooks/usePanelStorage';
|
||||||
@ -21,14 +19,8 @@ import { WorkflowName } from './WorkflowName';
|
|||||||
|
|
||||||
const panelGroupStyles: CSSProperties = { height: '100%', width: '100%' };
|
const panelGroupStyles: CSSProperties = { height: '100%', width: '100%' };
|
||||||
|
|
||||||
const selector = createMemoizedSelector(selectWorkflowSlice, (workflow) => {
|
|
||||||
return {
|
|
||||||
mode: workflow.mode,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const NodeEditorPanelGroup = () => {
|
const NodeEditorPanelGroup = () => {
|
||||||
const { mode } = useAppSelector(selector);
|
const mode = useAppSelector((s) => s.workflow.mode);
|
||||||
const panelGroupRef = useRef<ImperativePanelGroupHandle>(null);
|
const panelGroupRef = useRef<ImperativePanelGroupHandle>(null);
|
||||||
const panelStorage = usePanelStorage();
|
const panelStorage = usePanelStorage();
|
||||||
|
|
||||||
|
@ -1,20 +1,12 @@
|
|||||||
import { Flex } from '@invoke-ai/ui-library';
|
import { Flex } from '@invoke-ai/ui-library';
|
||||||
import { createMemoizedSelector } from 'app/store/createMemoizedSelector';
|
|
||||||
import { useAppSelector } from 'app/store/storeHooks';
|
import { useAppSelector } from 'app/store/storeHooks';
|
||||||
import SaveWorkflowButton from 'features/nodes/components/flow/panels/TopPanel/SaveWorkflowButton';
|
import SaveWorkflowButton from 'features/nodes/components/flow/panels/TopPanel/SaveWorkflowButton';
|
||||||
import { selectWorkflowSlice } from 'features/nodes/store/workflowSlice';
|
|
||||||
import { NewWorkflowButton } from 'features/workflowLibrary/components/NewWorkflowButton';
|
import { NewWorkflowButton } from 'features/workflowLibrary/components/NewWorkflowButton';
|
||||||
|
|
||||||
import { ModeToggle } from './ModeToggle';
|
import { ModeToggle } from './ModeToggle';
|
||||||
|
|
||||||
const selector = createMemoizedSelector(selectWorkflowSlice, (workflow) => {
|
|
||||||
return {
|
|
||||||
mode: workflow.mode,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
export const WorkflowMenu = () => {
|
export const WorkflowMenu = () => {
|
||||||
const { mode } = useAppSelector(selector);
|
const mode = useAppSelector((s) => s.workflow.mode);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex gap="2" alignItems="center">
|
<Flex gap="2" alignItems="center">
|
||||||
|
@ -2,13 +2,15 @@ import { Box } from '@invoke-ai/ui-library';
|
|||||||
import { ControlLayersEditor } from 'features/controlLayers/components/ControlLayersEditor';
|
import { ControlLayersEditor } from 'features/controlLayers/components/ControlLayersEditor';
|
||||||
import { ImageComparisonDroppable } from 'features/gallery/components/ImageViewer/ImageComparisonDroppable';
|
import { ImageComparisonDroppable } from 'features/gallery/components/ImageViewer/ImageComparisonDroppable';
|
||||||
import { ImageViewer } from 'features/gallery/components/ImageViewer/ImageViewer';
|
import { ImageViewer } from 'features/gallery/components/ImageViewer/ImageViewer';
|
||||||
|
import { useImageViewer } from 'features/gallery/components/ImageViewer/useImageViewer';
|
||||||
import { memo } from 'react';
|
import { memo } from 'react';
|
||||||
|
|
||||||
const TextToImageTab = () => {
|
const TextToImageTab = () => {
|
||||||
|
const imageViewer = useImageViewer();
|
||||||
return (
|
return (
|
||||||
<Box layerStyle="first" position="relative" w="full" h="full" p={2} borderRadius="base">
|
<Box layerStyle="first" position="relative" w="full" h="full" p={2} borderRadius="base">
|
||||||
<ControlLayersEditor />
|
<ControlLayersEditor />
|
||||||
<ImageViewer />
|
{imageViewer.isOpen && <ImageViewer />}
|
||||||
<ImageComparisonDroppable />
|
<ImageComparisonDroppable />
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
|
Loading…
Reference in New Issue
Block a user