From 5e1a6ae3346c8e8cde06139e6783c9bbb3c4e008 Mon Sep 17 00:00:00 2001 From: psychedelicious <4822129+psychedelicious@users.noreply.github.com> Date: Sun, 16 Jun 2024 11:36:36 +1000 Subject: [PATCH] refactor(ui): fix delete image stuff --- .../listeners/boardAndImagesDeleted.ts | 28 +++--- .../listeners/imageDeletionListeners.ts | 94 +++++++------------ .../listeners/imageToDeleteSelected.ts | 7 +- .../controlLayers/store/canvasV2Slice.ts | 4 + .../store/controlAdaptersReducers.ts | 3 + .../controlLayers/store/ipAdaptersReducers.ts | 6 +- .../controlLayers/store/layersReducers.ts | 3 + .../controlLayers/store/regionsReducers.ts | 3 + .../components/DeleteImageModal.tsx | 23 ++--- .../components/ImageUsageMessage.tsx | 6 +- .../deleteImageModal/store/selectors.ts | 75 ++++----------- .../features/deleteImageModal/store/types.ts | 6 +- .../components/Boards/DeleteBoardModal.tsx | 27 ++---- .../SingleSelectionMenuItems.tsx | 2 - 14 files changed, 109 insertions(+), 178 deletions(-) diff --git a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/boardAndImagesDeleted.ts b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/boardAndImagesDeleted.ts index eb7a793d3a..d3f20971b7 100644 --- a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/boardAndImagesDeleted.ts +++ b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/boardAndImagesDeleted.ts @@ -1,7 +1,5 @@ import type { AppStartListening } from 'app/store/middleware/listenerMiddleware'; -import { resetCanvas } from 'features/canvas/store/canvasSlice'; -import { controlAdaptersReset } from 'features/controlAdapters/store/controlAdaptersSlice'; -import { allLayersDeleted } from 'features/controlLayers/store/canvasV2Slice'; +import { caAllDeleted, ipaAllDeleted, layerAllDeleted } from 'features/controlLayers/store/canvasV2Slice'; import { getImageUsage } from 'features/deleteImageModal/store/selectors'; import { nodeEditorReset } from 'features/nodes/store/nodesSlice'; import { imagesApi } from 'services/api/endpoints/images'; @@ -14,18 +12,18 @@ export const addDeleteBoardAndImagesFulfilledListener = (startAppListening: AppS // Remove all deleted images from the UI - let wasCanvasReset = false; + let wereLayersReset = false; let wasNodeEditorReset = false; let wereControlAdaptersReset = false; - let wereControlLayersReset = false; + let wereIPAdaptersReset = false; - const { canvas, nodes, controlAdapters, canvasV2 } = getState(); + const { nodes, canvasV2 } = getState(); deleted_images.forEach((image_name) => { - const imageUsage = getImageUsage(canvas, nodes.present, controlAdapters, canvasV2, image_name); + const imageUsage = getImageUsage(nodes.present, canvasV2, image_name); - if (imageUsage.isCanvasImage && !wasCanvasReset) { - dispatch(resetCanvas()); - wasCanvasReset = true; + if (imageUsage.isLayerImage && !wereLayersReset) { + dispatch(layerAllDeleted()); + wereLayersReset = true; } if (imageUsage.isNodesImage && !wasNodeEditorReset) { @@ -33,14 +31,14 @@ export const addDeleteBoardAndImagesFulfilledListener = (startAppListening: AppS wasNodeEditorReset = true; } - if (imageUsage.isControlImage && !wereControlAdaptersReset) { - dispatch(controlAdaptersReset()); + if (imageUsage.isControlAdapterImage && !wereControlAdaptersReset) { + dispatch(caAllDeleted()); wereControlAdaptersReset = true; } - if (imageUsage.isControlLayerImage && !wereControlLayersReset) { - dispatch(allLayersDeleted()); - wereControlLayersReset = true; + if (imageUsage.isIPAdapterImage && !wereIPAdaptersReset) { + dispatch(ipaAllDeleted()); + wereIPAdaptersReset = true; } }); }, diff --git a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageDeletionListeners.ts b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageDeletionListeners.ts index cb4e9ec8c8..7b579d2a4a 100644 --- a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageDeletionListeners.ts +++ b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageDeletionListeners.ts @@ -3,18 +3,11 @@ import type { AppStartListening } from 'app/store/middleware/listenerMiddleware' import type { AppDispatch, RootState } from 'app/store/store'; import { resetCanvas } from 'features/canvas/store/canvasSlice'; import { - controlAdapterImageChanged, - controlAdapterProcessedImageChanged, - selectControlAdapterAll, -} from 'features/controlAdapters/store/controlAdaptersSlice'; -import { isControlNetOrT2IAdapter } from 'features/controlAdapters/store/types'; -import { layerDeleted } from 'features/controlLayers/store/canvasV2Slice'; -import { - isControlAdapterLayer, - isInitialImageLayer, - isIPAdapterLayer, - isRegionalGuidanceLayer, -} from 'features/controlLayers/store/types'; + caImageChanged, + caProcessedImageChanged, + ipaImageChanged, + layerDeleted, +} from 'features/controlLayers/store/canvasV2Slice'; import { imageDeletionConfirmed } from 'features/deleteImageModal/store/actions'; import { isModalOpenChanged } from 'features/deleteImageModal/store/slice'; import { selectListImagesQueryArgs } from 'features/gallery/store/gallerySelectors'; @@ -48,51 +41,33 @@ const deleteNodesImages = (state: RootState, dispatch: AppDispatch, imageDTO: Im }; const deleteControlAdapterImages = (state: RootState, dispatch: AppDispatch, imageDTO: ImageDTO) => { - forEach(selectControlAdapterAll(state.controlAdapters), (ca) => { - if ( - ca.controlImage === imageDTO.image_name || - (isControlNetOrT2IAdapter(ca) && ca.processedControlImage === imageDTO.image_name) - ) { - dispatch( - controlAdapterImageChanged({ - id: ca.id, - controlImage: null, - }) - ); - dispatch( - controlAdapterProcessedImageChanged({ - id: ca.id, - processedControlImage: null, - }) - ); + state.canvasV2.controlAdapters.forEach(({ id, image, processedImage }) => { + if (image?.name === imageDTO.image_name || processedImage?.name === imageDTO.image_name) { + dispatch(caImageChanged({ id, imageDTO: null })); + dispatch(caProcessedImageChanged({ id, imageDTO: null })); } }); }; -const deleteControlLayerImages = (state: RootState, dispatch: AppDispatch, imageDTO: ImageDTO) => { - state.canvasV2.layers.forEach((l) => { - if (isRegionalGuidanceLayer(l)) { - if (l.ipAdapters.some((ipa) => ipa.image?.name === imageDTO.image_name)) { - dispatch(layerDeleted(l.id)); +const deleteIPAdapterImages = (state: RootState, dispatch: AppDispatch, imageDTO: ImageDTO) => { + state.canvasV2.ipAdapters.forEach(({ id, image }) => { + if (image?.name === imageDTO.image_name) { + dispatch(ipaImageChanged({ id, imageDTO: null })); + } + }); +}; + +const deleteLayerImages = (state: RootState, dispatch: AppDispatch, imageDTO: ImageDTO) => { + state.canvasV2.layers.forEach(({ id, objects }) => { + let shouldDelete = false; + for (const obj of objects) { + if (obj.type === 'image' && obj.image.name === imageDTO.image_name) { + shouldDelete = true; + break; } } - if (isControlAdapterLayer(l)) { - if ( - l.controlAdapter.image?.name === imageDTO.image_name || - l.controlAdapter.processedImage?.name === imageDTO.image_name - ) { - dispatch(layerDeleted(l.id)); - } - } - if (isIPAdapterLayer(l)) { - if (l.ipAdapter.image?.name === imageDTO.image_name) { - dispatch(layerDeleted(l.id)); - } - } - if (isInitialImageLayer(l)) { - if (l.image?.name === imageDTO.image_name) { - dispatch(layerDeleted(l.id)); - } + if (shouldDelete) { + dispatch(layerDeleted({ id })); } }); }; @@ -145,14 +120,10 @@ export const addImageDeletionListeners = (startAppListening: AppStartListening) } } - // We need to reset the features where the image is in use - none of these work if their image(s) don't exist - if (imageUsage.isCanvasImage) { - dispatch(resetCanvas()); - } - - deleteControlAdapterImages(state, dispatch, imageDTO); deleteNodesImages(state, dispatch, imageDTO); - deleteControlLayerImages(state, dispatch, imageDTO); + deleteControlAdapterImages(state, dispatch, imageDTO); + deleteIPAdapterImages(state, dispatch, imageDTO); + deleteLayerImages(state, dispatch, imageDTO); } catch { // no-op } finally { @@ -189,14 +160,15 @@ export const addImageDeletionListeners = (startAppListening: AppStartListening) // We need to reset the features where the image is in use - none of these work if their image(s) don't exist - if (imagesUsage.some((i) => i.isCanvasImage)) { + if (imagesUsage.some((i) => i.isLayerImage)) { dispatch(resetCanvas()); } imageDTOs.forEach((imageDTO) => { - deleteControlAdapterImages(state, dispatch, imageDTO); deleteNodesImages(state, dispatch, imageDTO); - deleteControlLayerImages(state, dispatch, imageDTO); + deleteControlAdapterImages(state, dispatch, imageDTO); + deleteIPAdapterImages(state, dispatch, imageDTO); + deleteLayerImages(state, dispatch, imageDTO); }); } catch { // no-op diff --git a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageToDeleteSelected.ts b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageToDeleteSelected.ts index 845c9a21f2..2e20d97b46 100644 --- a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageToDeleteSelected.ts +++ b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/imageToDeleteSelected.ts @@ -13,9 +13,10 @@ export const addImageToDeleteSelectedListener = (startAppListening: AppStartList const imagesUsage = selectImageUsage(getState()); const isImageInUse = - imagesUsage.some((i) => i.isCanvasImage) || - imagesUsage.some((i) => i.isControlImage) || - imagesUsage.some((i) => i.isNodesImage); + imagesUsage.some((i) => i.isLayerImage) || + imagesUsage.some((i) => i.isControlAdapterImage) || + imagesUsage.some((i) => i.isIPAdapterImage) || + imagesUsage.some((i) => i.isLayerImage); if (shouldConfirmOnDelete || isImageInUse) { dispatch(isModalOpenChanged(true)); diff --git a/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts b/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts index 8b1721cb3f..60abd47769 100644 --- a/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts +++ b/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts @@ -194,11 +194,13 @@ export const { layerLinePointAdded, layerRectAdded, layerImageAdded, + layerAllDeleted, // IP Adapters ipaAdded, ipaRecalled, ipaIsEnabledToggled, ipaDeleted, + ipaAllDeleted, ipaImageChanged, ipaMethodChanged, ipaModelChanged, @@ -209,6 +211,7 @@ export const { caAdded, caBboxChanged, caDeleted, + caAllDeleted, caIsEnabledToggled, caMovedBackwardOne, caMovedForwardOne, @@ -234,6 +237,7 @@ export const { rgTranslated, rgBboxChanged, rgDeleted, + rgAllDeleted, rgGlobalOpacityChanged, rgMovedForwardOne, rgMovedToFront, diff --git a/invokeai/frontend/web/src/features/controlLayers/store/controlAdaptersReducers.ts b/invokeai/frontend/web/src/features/controlLayers/store/controlAdaptersReducers.ts index 84c12e5f3a..0a744c6c28 100644 --- a/invokeai/frontend/web/src/features/controlLayers/store/controlAdaptersReducers.ts +++ b/invokeai/frontend/web/src/features/controlLayers/store/controlAdaptersReducers.ts @@ -82,6 +82,9 @@ export const controlAdaptersReducers = { const { id } = action.payload; state.controlAdapters = state.controlAdapters.filter((ca) => ca.id !== id); }, + caAllDeleted: (state) => { + state.controlAdapters = []; + }, caOpacityChanged: (state, action: PayloadAction<{ id: string; opacity: number }>) => { const { id, opacity } = action.payload; const ca = selectCA(state, id); diff --git a/invokeai/frontend/web/src/features/controlLayers/store/ipAdaptersReducers.ts b/invokeai/frontend/web/src/features/controlLayers/store/ipAdaptersReducers.ts index 16cf222b60..fb6ff8d987 100644 --- a/invokeai/frontend/web/src/features/controlLayers/store/ipAdaptersReducers.ts +++ b/invokeai/frontend/web/src/features/controlLayers/store/ipAdaptersReducers.ts @@ -39,7 +39,11 @@ export const ipAdaptersReducers = { } }, ipaDeleted: (state, action: PayloadAction<{ id: string }>) => { - state.ipAdapters = state.ipAdapters.filter((ipa) => ipa.id !== action.payload.id); + const { id } = action.payload; + state.ipAdapters = state.ipAdapters.filter((ipa) => ipa.id !== id); + }, + ipaAllDeleted: (state) => { + state.ipAdapters = []; }, ipaImageChanged: (state, action: PayloadAction<{ id: string; imageDTO: ImageDTO | null }>) => { const { id, imageDTO } = action.payload; diff --git a/invokeai/frontend/web/src/features/controlLayers/store/layersReducers.ts b/invokeai/frontend/web/src/features/controlLayers/store/layersReducers.ts index d44585fd4f..6b3e58f8a2 100644 --- a/invokeai/frontend/web/src/features/controlLayers/store/layersReducers.ts +++ b/invokeai/frontend/web/src/features/controlLayers/store/layersReducers.ts @@ -88,6 +88,9 @@ export const layersReducers = { const { id } = action.payload; state.layers = state.layers.filter((l) => l.id !== id); }, + layerAllDeleted: (state) => { + state.layers = []; + }, layerOpacityChanged: (state, action: PayloadAction<{ id: string; opacity: number }>) => { const { id, opacity } = action.payload; const layer = selectLayer(state, id); diff --git a/invokeai/frontend/web/src/features/controlLayers/store/regionsReducers.ts b/invokeai/frontend/web/src/features/controlLayers/store/regionsReducers.ts index 56f3b935d9..35b584adb7 100644 --- a/invokeai/frontend/web/src/features/controlLayers/store/regionsReducers.ts +++ b/invokeai/frontend/web/src/features/controlLayers/store/regionsReducers.ts @@ -117,6 +117,9 @@ export const regionsReducers = { const { id } = action.payload; state.regions = state.regions.filter((ca) => ca.id !== id); }, + rgAllDeleted: (state) => { + state.regions = []; + }, rgGlobalOpacityChanged: (state, action: PayloadAction<{ opacity: number }>) => { const { opacity } = action.payload; state.maskFillOpacity = opacity; diff --git a/invokeai/frontend/web/src/features/deleteImageModal/components/DeleteImageModal.tsx b/invokeai/frontend/web/src/features/deleteImageModal/components/DeleteImageModal.tsx index 3d33999c71..30f59aed1f 100644 --- a/invokeai/frontend/web/src/features/deleteImageModal/components/DeleteImageModal.tsx +++ b/invokeai/frontend/web/src/features/deleteImageModal/components/DeleteImageModal.tsx @@ -1,8 +1,6 @@ import { ConfirmationAlertDialog, Divider, Flex, FormControl, FormLabel, Switch, Text } from '@invoke-ai/ui-library'; import { createMemoizedSelector } from 'app/store/createMemoizedSelector'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; -import { selectCanvasSlice } from 'features/canvas/store/canvasSlice'; -import { selectControlAdaptersSlice } from 'features/controlAdapters/store/controlAdaptersSlice'; import { selectCanvasV2Slice } from 'features/controlLayers/store/canvasV2Slice'; import { imageDeletionConfirmed } from 'features/deleteImageModal/store/actions'; import { getImageUsage, selectImageUsage } from 'features/deleteImageModal/store/selectors'; @@ -22,26 +20,17 @@ import { useTranslation } from 'react-i18next'; import ImageUsageMessage from './ImageUsageMessage'; const selectImageUsages = createMemoizedSelector( - [ - selectDeleteImageModalSlice, - selectCanvasSlice, - selectNodesSlice, - selectControlAdaptersSlice, - selectCanvasV2Slice, - selectImageUsage, - ], - (deleteImageModal, canvas, nodes, controlAdapters, controlLayers, imagesUsage) => { + [selectDeleteImageModalSlice, selectNodesSlice, selectCanvasV2Slice, selectImageUsage], + (deleteImageModal, nodes, canvasV2, imagesUsage) => { const { imagesToDelete } = deleteImageModal; - const allImageUsage = (imagesToDelete ?? []).map(({ image_name }) => - getImageUsage(canvas, nodes, controlAdapters, canvasV2, image_name) - ); + const allImageUsage = (imagesToDelete ?? []).map(({ image_name }) => getImageUsage(nodes, canvasV2, image_name)); const imageUsageSummary: ImageUsage = { - isCanvasImage: some(allImageUsage, (i) => i.isCanvasImage), + isLayerImage: some(allImageUsage, (i) => i.isLayerImage), isNodesImage: some(allImageUsage, (i) => i.isNodesImage), - isControlImage: some(allImageUsage, (i) => i.isControlImage), - isControlLayerImage: some(allImageUsage, (i) => i.isControlLayerImage), + isControlAdapterImage: some(allImageUsage, (i) => i.isControlAdapterImage), + isIPAdapterImage: some(allImageUsage, (i) => i.isIPAdapterImage), }; return { diff --git a/invokeai/frontend/web/src/features/deleteImageModal/components/ImageUsageMessage.tsx b/invokeai/frontend/web/src/features/deleteImageModal/components/ImageUsageMessage.tsx index d76716d01d..8f71922e0a 100644 --- a/invokeai/frontend/web/src/features/deleteImageModal/components/ImageUsageMessage.tsx +++ b/invokeai/frontend/web/src/features/deleteImageModal/components/ImageUsageMessage.tsx @@ -29,10 +29,10 @@ const ImageUsageMessage = (props: Props) => { <> {topMessage} - {imageUsage.isCanvasImage && {t('ui.tabs.canvasTab')}} - {imageUsage.isControlImage && {t('common.controlNet')}} + {imageUsage.isLayerImage && {t('controlLayers.layers')}} + {imageUsage.isControlAdapterImage && {t('controlLayers.controlAdapters')}} + {imageUsage.isIPAdapterImage && {t('controlLayers.ipAdapters')}} {imageUsage.isNodesImage && {t('ui.tabs.workflowsTab')}} - {imageUsage.isControlLayerImage && {t('ui.tabs.generationTab')}} {bottomMessage} diff --git a/invokeai/frontend/web/src/features/deleteImageModal/store/selectors.ts b/invokeai/frontend/web/src/features/deleteImageModal/store/selectors.ts index ce36e9080d..5874fe9c71 100644 --- a/invokeai/frontend/web/src/features/deleteImageModal/store/selectors.ts +++ b/invokeai/frontend/web/src/features/deleteImageModal/store/selectors.ts @@ -1,20 +1,6 @@ import { createMemoizedSelector } from 'app/store/createMemoizedSelector'; -import { selectCanvasSlice } from 'features/canvas/store/canvasSlice'; -import type { CanvasState } from 'features/canvas/store/canvasTypes'; -import { - selectControlAdapterAll, - selectControlAdaptersSlice, -} from 'features/controlAdapters/store/controlAdaptersSlice'; -import type { ControlAdaptersState } from 'features/controlAdapters/store/types'; -import { isControlNetOrT2IAdapter } from 'features/controlAdapters/store/types'; import { selectCanvasV2Slice } from 'features/controlLayers/store/canvasV2Slice'; import type { CanvasV2State } from 'features/controlLayers/store/types'; -import { - isControlAdapterLayer, - isInitialImageLayer, - isIPAdapterLayer, - isRegionalGuidanceLayer, -} from 'features/controlLayers/store/types'; import { selectDeleteImageModalSlice } from 'features/deleteImageModal/store/slice'; import { selectNodesSlice } from 'features/nodes/store/nodesSlice'; import type { NodesState } from 'features/nodes/store/types'; @@ -24,47 +10,28 @@ import { some } from 'lodash-es'; import type { ImageUsage } from './types'; -export const getImageUsage = ( - canvas: CanvasState, - nodes: NodesState, - controlAdapters: ControlAdaptersState, - controlLayers: CanvasV2State, - image_name: string -) => { - const isCanvasImage = canvas.layerState.objects.some((obj) => obj.kind === 'image' && obj.imageName === image_name); - - const isNodesImage = nodes.nodes.filter(isInvocationNode).some((node) => { - return some( - node.data.inputs, - (input) => isImageFieldInputInstance(input) && input.value?.image_name === image_name - ); - }); - - const isControlImage = selectControlAdapterAll(controlAdapters).some( - (ca) => ca.controlImage === image_name || (isControlNetOrT2IAdapter(ca) && ca.processedControlImage === image_name) +export const getImageUsage = (nodes: NodesState, canvasV2: CanvasV2State, image_name: string) => { + const isLayerImage = canvasV2.layers.some((layer) => + layer.objects.some((obj) => obj.type === 'image' && obj.image.name === image_name) ); - const isControlLayerImage = controlLayers.layers.some((l) => { - if (isRegionalGuidanceLayer(l)) { - return l.ipAdapters.some((ipa) => ipa.image?.name === image_name); - } - if (isControlAdapterLayer(l)) { - return l.controlAdapter.image?.name === image_name || l.controlAdapter.processedImage?.name === image_name; - } - if (isIPAdapterLayer(l)) { - return l.ipAdapter.image?.name === image_name; - } - if (isInitialImageLayer(l)) { - return l.image?.name === image_name; - } - return false; - }); + const isNodesImage = nodes.nodes + .filter(isInvocationNode) + .some((node) => + some(node.data.inputs, (input) => isImageFieldInputInstance(input) && input.value?.image_name === image_name) + ); + + const isControlAdapterImage = canvasV2.controlAdapters.some( + (ca) => ca.image?.name === image_name || ca.processedImage?.name === image_name + ); + + const isIPAdapterImage = canvasV2.ipAdapters.some((ipa) => ipa.image?.name === image_name); const imageUsage: ImageUsage = { - isCanvasImage, + isLayerImage, isNodesImage, - isControlImage, - isControlLayerImage, + isControlAdapterImage, + isIPAdapterImage, }; return imageUsage; @@ -72,20 +39,16 @@ export const getImageUsage = ( export const selectImageUsage = createMemoizedSelector( selectDeleteImageModalSlice, - selectCanvasSlice, selectNodesSlice, - selectControlAdaptersSlice, selectCanvasV2Slice, - (deleteImageModal, canvas, nodes, controlAdapters, controlLayers) => { + (deleteImageModal, nodes, canvasV2) => { const { imagesToDelete } = deleteImageModal; if (!imagesToDelete.length) { return []; } - const imagesUsage = imagesToDelete.map((i) => - getImageUsage(canvas, nodes, controlAdapters, canvasV2, i.image_name) - ); + const imagesUsage = imagesToDelete.map((i) => getImageUsage(nodes, canvasV2, i.image_name)); return imagesUsage; } diff --git a/invokeai/frontend/web/src/features/deleteImageModal/store/types.ts b/invokeai/frontend/web/src/features/deleteImageModal/store/types.ts index 2cc3dd90b4..259a510646 100644 --- a/invokeai/frontend/web/src/features/deleteImageModal/store/types.ts +++ b/invokeai/frontend/web/src/features/deleteImageModal/store/types.ts @@ -6,8 +6,8 @@ export type DeleteImageState = { }; export type ImageUsage = { - isCanvasImage: boolean; isNodesImage: boolean; - isControlImage: boolean; - isControlLayerImage: boolean; + isControlAdapterImage: boolean; + isIPAdapterImage: boolean; + isLayerImage: boolean; }; diff --git a/invokeai/frontend/web/src/features/gallery/components/Boards/DeleteBoardModal.tsx b/invokeai/frontend/web/src/features/gallery/components/Boards/DeleteBoardModal.tsx index fd9a52a69d..9ef21177da 100644 --- a/invokeai/frontend/web/src/features/gallery/components/Boards/DeleteBoardModal.tsx +++ b/invokeai/frontend/web/src/features/gallery/components/Boards/DeleteBoardModal.tsx @@ -13,8 +13,6 @@ import { import { skipToken } from '@reduxjs/toolkit/query'; import { createMemoizedSelector } from 'app/store/createMemoizedSelector'; import { useAppSelector } from 'app/store/storeHooks'; -import { selectCanvasSlice } from 'features/canvas/store/canvasSlice'; -import { selectControlAdaptersSlice } from 'features/controlAdapters/store/controlAdaptersSlice'; import { selectCanvasV2Slice } from 'features/controlLayers/store/canvasV2Slice'; import ImageUsageMessage from 'features/deleteImageModal/components/ImageUsageMessage'; import { getImageUsage } from 'features/deleteImageModal/store/selectors'; @@ -41,23 +39,18 @@ const DeleteBoardModal = (props: Props) => { const selectImageUsageSummary = useMemo( () => - createMemoizedSelector( - [selectCanvasSlice, selectNodesSlice, selectControlAdaptersSlice, selectCanvasV2Slice], - (canvas, nodes, controlAdapters, controlLayers) => { - const allImageUsage = (boardImageNames ?? []).map((imageName) => - getImageUsage(canvas, nodes, controlAdapters, canvasV2, imageName) - ); + createMemoizedSelector([selectNodesSlice, selectCanvasV2Slice], (nodes, canvasV2) => { + const allImageUsage = (boardImageNames ?? []).map((imageName) => getImageUsage(nodes, canvasV2, imageName)); - const imageUsageSummary: ImageUsage = { - isCanvasImage: some(allImageUsage, (i) => i.isCanvasImage), - isNodesImage: some(allImageUsage, (i) => i.isNodesImage), - isControlImage: some(allImageUsage, (i) => i.isControlImage), - isControlLayerImage: some(allImageUsage, (i) => i.isControlLayerImage), - }; + const imageUsageSummary: ImageUsage = { + isLayerImage: some(allImageUsage, (i) => i.isLayerImage), + isNodesImage: some(allImageUsage, (i) => i.isNodesImage), + isControlAdapterImage: some(allImageUsage, (i) => i.isControlAdapterImage), + isIPAdapterImage: some(allImageUsage, (i) => i.isIPAdapterImage), + }; - return imageUsageSummary; - } - ), + return imageUsageSummary; + }), [boardImageNames] ); diff --git a/invokeai/frontend/web/src/features/gallery/components/ImageContextMenu/SingleSelectionMenuItems.tsx b/invokeai/frontend/web/src/features/gallery/components/ImageContextMenu/SingleSelectionMenuItems.tsx index 6af1aaf8b2..19ad53eede 100644 --- a/invokeai/frontend/web/src/features/gallery/components/ImageContextMenu/SingleSelectionMenuItems.tsx +++ b/invokeai/frontend/web/src/features/gallery/components/ImageContextMenu/SingleSelectionMenuItems.tsx @@ -15,7 +15,6 @@ import { imageToCompareChanged } from 'features/gallery/store/gallerySlice'; import { $templates } from 'features/nodes/store/nodesSlice'; import { selectOptimalDimension } from 'features/parameters/store/generationSlice'; import { upscaleInitialImageChanged } from 'features/parameters/store/upscaleSlice'; -import { useFeatureStatus } from 'features/system/hooks/useFeatureStatus'; import { toast } from 'features/toast/toast'; import { setActiveTab } from 'features/ui/store/uiSlice'; import { useGetAndLoadEmbeddedWorkflow } from 'features/workflowLibrary/hooks/useGetAndLoadEmbeddedWorkflow'; @@ -52,7 +51,6 @@ const SingleSelectionMenuItems = (props: SingleSelectionMenuItemsProps) => { const maySelectForCompare = useAppSelector((s) => s.gallery.imageToCompare?.image_name !== imageDTO.image_name); const dispatch = useAppDispatch(); const { t } = useTranslation(); - const isCanvasEnabled = useFeatureStatus('canvas'); const customStarUi = useStore($customStarUI); const { downloadImage } = useDownloadImage(); const templates = useStore($templates);