mirror of
https://github.com/invoke-ai/InvokeAI.git
synced 2026-04-23 03:00:31 -04:00
- Add `Open in Viewer` - Remove `Send to Image to Image` - Fix `Send to Canvas` - Split out logic for composability
174 lines
7.3 KiB
TypeScript
174 lines
7.3 KiB
TypeScript
import { ButtonGroup, IconButton, Menu, MenuButton, MenuList } from '@invoke-ai/ui-library';
|
|
import { useStore } from '@nanostores/react';
|
|
import { skipToken } from '@reduxjs/toolkit/query';
|
|
import { $isConnected } from 'app/hooks/useSocketIO';
|
|
import { adHocPostProcessingRequested } from 'app/store/middleware/listenerMiddleware/listeners/addAdHocPostProcessingRequestedListener';
|
|
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
|
import { INTERACTION_SCOPES } from 'common/hooks/interactionScopes';
|
|
import { DeleteImageButton } from 'features/deleteImageModal/components/DeleteImageButton';
|
|
import { imagesToDeleteSelected } from 'features/deleteImageModal/store/slice';
|
|
import SingleSelectionMenuItems from 'features/gallery/components/ImageContextMenu/SingleSelectionMenuItems';
|
|
import { useImageActions } from 'features/gallery/hooks/useImageActions';
|
|
import { selectLastSelectedImage } from 'features/gallery/store/gallerySelectors';
|
|
import { parseAndRecallImageDimensions } from 'features/metadata/util/handlers';
|
|
import { $templates } from 'features/nodes/store/nodesSlice';
|
|
import { PostProcessingPopover } from 'features/parameters/components/PostProcessing/PostProcessingPopover';
|
|
import { useIsQueueMutationInProgress } from 'features/queue/hooks/useIsQueueMutationInProgress';
|
|
import { useFeatureStatus } from 'features/system/hooks/useFeatureStatus';
|
|
import { useGetAndLoadEmbeddedWorkflow } from 'features/workflowLibrary/hooks/useGetAndLoadEmbeddedWorkflow';
|
|
import { size } from 'lodash-es';
|
|
import { memo, useCallback, useMemo } from 'react';
|
|
import { useHotkeys } from 'react-hotkeys-hook';
|
|
import { useTranslation } from 'react-i18next';
|
|
import {
|
|
PiArrowsCounterClockwiseBold,
|
|
PiAsteriskBold,
|
|
PiDotsThreeOutlineFill,
|
|
PiFlowArrowBold,
|
|
PiPlantBold,
|
|
PiQuotesBold,
|
|
PiRulerBold,
|
|
} from 'react-icons/pi';
|
|
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
|
import { $progressImage } from 'services/events/setEventListeners';
|
|
|
|
const CurrentImageButtons = () => {
|
|
const dispatch = useAppDispatch();
|
|
const isConnected = useStore($isConnected);
|
|
const lastSelectedImage = useAppSelector(selectLastSelectedImage);
|
|
const progressImage = useStore($progressImage);
|
|
const selection = useAppSelector((s) => s.gallery.selection);
|
|
const shouldDisableToolbarButtons = useMemo(() => {
|
|
return Boolean(progressImage) || !lastSelectedImage;
|
|
}, [lastSelectedImage, progressImage]);
|
|
const templates = useStore($templates);
|
|
const isUpscalingEnabled = useFeatureStatus('upscaling');
|
|
const isQueueMutationInProgress = useIsQueueMutationInProgress();
|
|
const { t } = useTranslation();
|
|
const isImageViewerActive = useStore(INTERACTION_SCOPES.imageViewer.$isActive);
|
|
const { currentData: imageDTO } = useGetImageDTOQuery(lastSelectedImage?.image_name ?? skipToken);
|
|
|
|
const { recallAll, remix, recallSeed, recallPrompts, hasMetadata, hasSeed, hasPrompts, isLoadingMetadata } =
|
|
useImageActions(lastSelectedImage?.image_name);
|
|
|
|
const { getAndLoadEmbeddedWorkflow, getAndLoadEmbeddedWorkflowResult } = useGetAndLoadEmbeddedWorkflow({});
|
|
|
|
const handleLoadWorkflow = useCallback(() => {
|
|
if (!lastSelectedImage || !lastSelectedImage.has_workflow) {
|
|
return;
|
|
}
|
|
getAndLoadEmbeddedWorkflow(lastSelectedImage.image_name);
|
|
}, [getAndLoadEmbeddedWorkflow, lastSelectedImage]);
|
|
|
|
const handleUseSize = useCallback(() => {
|
|
parseAndRecallImageDimensions(lastSelectedImage);
|
|
}, [lastSelectedImage]);
|
|
const handleClickUpscale = useCallback(() => {
|
|
if (!imageDTO) {
|
|
return;
|
|
}
|
|
dispatch(adHocPostProcessingRequested({ imageDTO }));
|
|
}, [dispatch, imageDTO]);
|
|
|
|
const handleDelete = useCallback(() => {
|
|
if (!imageDTO) {
|
|
return;
|
|
}
|
|
dispatch(imagesToDeleteSelected(selection));
|
|
}, [dispatch, imageDTO, selection]);
|
|
|
|
useHotkeys('w', handleLoadWorkflow, { enabled: isImageViewerActive }, [lastSelectedImage, isImageViewerActive]);
|
|
useHotkeys('a', recallAll, { enabled: isImageViewerActive }, [recallAll, isImageViewerActive]);
|
|
useHotkeys('s', recallSeed, { enabled: isImageViewerActive }, [recallSeed, isImageViewerActive]);
|
|
useHotkeys('p', recallPrompts, { enabled: isImageViewerActive }, [recallPrompts, isImageViewerActive]);
|
|
useHotkeys('r', remix, { enabled: isImageViewerActive }, [remix, isImageViewerActive]);
|
|
useHotkeys('d', handleUseSize, { enabled: isImageViewerActive }, [handleUseSize, isImageViewerActive]);
|
|
useHotkeys(
|
|
'shift+u',
|
|
handleClickUpscale,
|
|
{ enabled: Boolean(isUpscalingEnabled && isImageViewerActive && isConnected) },
|
|
[isUpscalingEnabled, imageDTO, shouldDisableToolbarButtons, isConnected, isImageViewerActive]
|
|
);
|
|
|
|
useHotkeys(['delete', 'backspace'], handleDelete, { enabled: isImageViewerActive }, [imageDTO, isImageViewerActive]);
|
|
|
|
return (
|
|
<>
|
|
<ButtonGroup isDisabled={shouldDisableToolbarButtons}>
|
|
<Menu isLazy>
|
|
<MenuButton
|
|
as={IconButton}
|
|
aria-label={t('parameters.imageActions')}
|
|
tooltip={t('parameters.imageActions')}
|
|
isDisabled={!imageDTO}
|
|
icon={<PiDotsThreeOutlineFill />}
|
|
/>
|
|
<MenuList>{imageDTO && <SingleSelectionMenuItems imageDTO={imageDTO} />}</MenuList>
|
|
</Menu>
|
|
</ButtonGroup>
|
|
|
|
<ButtonGroup isDisabled={shouldDisableToolbarButtons}>
|
|
<IconButton
|
|
icon={<PiFlowArrowBold />}
|
|
tooltip={`${t('nodes.loadWorkflow')} (W)`}
|
|
aria-label={`${t('nodes.loadWorkflow')} (W)`}
|
|
isDisabled={!imageDTO?.has_workflow || !size(templates)}
|
|
onClick={handleLoadWorkflow}
|
|
isLoading={getAndLoadEmbeddedWorkflowResult.isLoading}
|
|
/>
|
|
<IconButton
|
|
isLoading={isLoadingMetadata}
|
|
icon={<PiArrowsCounterClockwiseBold />}
|
|
tooltip={`${t('parameters.remixImage')} (R)`}
|
|
aria-label={`${t('parameters.remixImage')} (R)`}
|
|
isDisabled={!hasMetadata}
|
|
onClick={remix}
|
|
/>
|
|
<IconButton
|
|
isLoading={isLoadingMetadata}
|
|
icon={<PiQuotesBold />}
|
|
tooltip={`${t('parameters.usePrompt')} (P)`}
|
|
aria-label={`${t('parameters.usePrompt')} (P)`}
|
|
isDisabled={!hasPrompts}
|
|
onClick={recallPrompts}
|
|
/>
|
|
<IconButton
|
|
isLoading={isLoadingMetadata}
|
|
icon={<PiPlantBold />}
|
|
tooltip={`${t('parameters.useSeed')} (S)`}
|
|
aria-label={`${t('parameters.useSeed')} (S)`}
|
|
isDisabled={!hasSeed}
|
|
onClick={recallSeed}
|
|
/>
|
|
<IconButton
|
|
isLoading={isLoadingMetadata}
|
|
icon={<PiRulerBold />}
|
|
tooltip={`${t('parameters.useSize')} (D)`}
|
|
aria-label={`${t('parameters.useSize')} (D)`}
|
|
onClick={handleUseSize}
|
|
/>
|
|
<IconButton
|
|
isLoading={isLoadingMetadata}
|
|
icon={<PiAsteriskBold />}
|
|
tooltip={`${t('parameters.useAll')} (A)`}
|
|
aria-label={`${t('parameters.useAll')} (A)`}
|
|
isDisabled={!hasMetadata}
|
|
onClick={recallAll}
|
|
/>
|
|
</ButtonGroup>
|
|
|
|
{isUpscalingEnabled && (
|
|
<ButtonGroup isDisabled={isQueueMutationInProgress}>
|
|
{isUpscalingEnabled && <PostProcessingPopover imageDTO={imageDTO} />}
|
|
</ButtonGroup>
|
|
)}
|
|
|
|
<ButtonGroup>
|
|
<DeleteImageButton onClick={handleDelete} />
|
|
</ButtonGroup>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default memo(CurrentImageButtons);
|