diff --git a/invokeai/frontend/web/public/locales/en.json b/invokeai/frontend/web/public/locales/en.json
index 83195488c5..cfeeef8d51 100644
--- a/invokeai/frontend/web/public/locales/en.json
+++ b/invokeai/frontend/web/public/locales/en.json
@@ -1645,6 +1645,11 @@
"storeNotInitialized": "Store is not initialized"
},
"controlLayers": {
+ "generateMode": "Generate",
+ "generateModeDesc": "Create individual images. Generated images are added directly to the gallery.",
+ "composeMode": "Compose",
+ "composeModeDesc": "Compose your work iterative. Generated images are added back to the canvas.",
+ "autoSave": "Auto-save to Gallery",
"resetCanvas": "Reset Canvas",
"resetAll": "Reset All",
"clearCaches": "Clear Caches",
diff --git a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/enqueueRequestedLinear.ts b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/enqueueRequestedLinear.ts
index d9cfae607a..9111eba123 100644
--- a/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/enqueueRequestedLinear.ts
+++ b/invokeai/frontend/web/src/app/store/middleware/listenerMiddleware/listeners/enqueueRequestedLinear.ts
@@ -21,7 +21,7 @@ export const addEnqueueRequestedLinear = (startAppListening: AppStartListening)
assert(manager, 'No model found in state');
let didStartStaging = false;
- if (!state.canvasV2.session.isStaging) {
+ if (!state.canvasV2.session.isStaging && state.canvasV2.session.mode === 'compose') {
dispatch(sessionStartedStaging());
didStartStaging = true;
}
@@ -50,7 +50,6 @@ export const addEnqueueRequestedLinear = (startAppListening: AppStartListening)
req.reset();
await req.unwrap();
} catch (error) {
- console.log('Error in enqueueRequestedLinear', error);
if (didStartStaging && getState().canvasV2.session.isStaging) {
dispatch(sessionStagingAreaReset());
}
diff --git a/invokeai/frontend/web/src/features/controlLayers/components/CanvasModeSwitcher.tsx b/invokeai/frontend/web/src/features/controlLayers/components/CanvasModeSwitcher.tsx
new file mode 100644
index 0000000000..81b9114153
--- /dev/null
+++ b/invokeai/frontend/web/src/features/controlLayers/components/CanvasModeSwitcher.tsx
@@ -0,0 +1,26 @@
+import { Button, ButtonGroup } from '@invoke-ai/ui-library';
+import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
+import { sessionModeChanged } from 'features/controlLayers/store/canvasV2Slice';
+import { memo, useCallback } from 'react';
+import { useTranslation } from 'react-i18next';
+
+export const CanvasModeSwitcher = memo(() => {
+ const { t } = useTranslation();
+ const dispatch = useAppDispatch();
+ const mode = useAppSelector((s) => s.canvasV2.session.mode);
+ const onClickGenerate = useCallback(() => dispatch(sessionModeChanged({ mode: 'generate' })), [dispatch]);
+ const onClickCompose = useCallback(() => dispatch(sessionModeChanged({ mode: 'compose' })), [dispatch]);
+
+ return (
+
+
+
+
+ );
+});
+
+CanvasModeSwitcher.displayName = 'CanvasModeSwitcher';
diff --git a/invokeai/frontend/web/src/features/controlLayers/components/CanvasResetViewButton.tsx b/invokeai/frontend/web/src/features/controlLayers/components/CanvasResetViewButton.tsx
index 756a31e837..11de7601a1 100644
--- a/invokeai/frontend/web/src/features/controlLayers/components/CanvasResetViewButton.tsx
+++ b/invokeai/frontend/web/src/features/controlLayers/components/CanvasResetViewButton.tsx
@@ -23,7 +23,7 @@ export const CanvasResetViewButton = memo(() => {
if (!canvasManager) {
return;
}
- canvasManager.stage.resetView();
+ canvasManager.stage.fitLayersToStage();
}, [canvasManager]);
const onReset = useCallback(() => {
diff --git a/invokeai/frontend/web/src/features/controlLayers/components/ControlLayersToolbar.tsx b/invokeai/frontend/web/src/features/controlLayers/components/ControlLayersToolbar.tsx
index ec968f19f8..5dcb801b4e 100644
--- a/invokeai/frontend/web/src/features/controlLayers/components/ControlLayersToolbar.tsx
+++ b/invokeai/frontend/web/src/features/controlLayers/components/ControlLayersToolbar.tsx
@@ -1,6 +1,7 @@
/* eslint-disable i18next/no-literal-string */
import { Flex, Spacer } from '@invoke-ai/ui-library';
import { useAppSelector } from 'app/store/storeHooks';
+import { CanvasModeSwitcher } from 'features/controlLayers/components/CanvasModeSwitcher';
import { CanvasResetViewButton } from 'features/controlLayers/components/CanvasResetViewButton';
import { CanvasScale } from 'features/controlLayers/components/CanvasScale';
import { CanvasSettingsPopover } from 'features/controlLayers/components/Settings/CanvasSettingsPopover';
@@ -23,11 +24,12 @@ export const ControlLayersToolbar = memo(() => {
{tool === 'brush' && }
{tool === 'eraser' && }
+
-
+
diff --git a/invokeai/frontend/web/src/features/controlLayers/konva/CanvasProgressImageModule.ts b/invokeai/frontend/web/src/features/controlLayers/konva/CanvasProgressImageModule.ts
index baaedd2c07..4d306c4a5c 100644
--- a/invokeai/frontend/web/src/features/controlLayers/konva/CanvasProgressImageModule.ts
+++ b/invokeai/frontend/web/src/features/controlLayers/konva/CanvasProgressImageModule.ts
@@ -70,13 +70,6 @@ export class CanvasProgressImageModule {
return;
}
- const { isStaging } = this.manager.stateApi.getSession();
-
- if (!isStaging) {
- release();
- return;
- }
-
this.isLoading = true;
const { x, y, width, height } = this.manager.stateApi.getBbox().rect;
diff --git a/invokeai/frontend/web/src/features/controlLayers/konva/CanvasStageModule.ts b/invokeai/frontend/web/src/features/controlLayers/konva/CanvasStageModule.ts
index 85504b527b..a88b48913b 100644
--- a/invokeai/frontend/web/src/features/controlLayers/konva/CanvasStageModule.ts
+++ b/invokeai/frontend/web/src/features/controlLayers/konva/CanvasStageModule.ts
@@ -33,6 +33,7 @@ export class CanvasStageModule {
const resizeObserver = new ResizeObserver(this.fitStageToContainer);
resizeObserver.observe(this.container);
this.fitStageToContainer();
+ this.fitLayersToStage();
return () => {
this.log.debug('Destroying stage');
@@ -91,10 +92,20 @@ export class CanvasStageModule {
}
};
- resetView() {
- this.log.trace('Resetting view');
- const { width, height } = this.getSize();
+ fitBboxToStage = () => {
+ this.log.trace('Fitting bbox to stage');
+ const bbox = this.manager.stateApi.getBbox();
+ this.fitRect(bbox.rect);
+ };
+
+ fitLayersToStage() {
+ this.log.trace('Fitting layers to stage');
const rect = this.getVisibleRect();
+ this.fitRect(rect);
+ }
+
+ fitRect = (rect: Rect) => {
+ const { width, height } = this.getSize();
const padding = 20; // Padding in absolute pixels
@@ -118,7 +129,7 @@ export class CanvasStageModule {
y,
scale,
});
- }
+ };
/**
* Gets the center of the stage in either absolute or relative coordinates
diff --git a/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts b/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts
index 7d6e19179f..688457d18f 100644
--- a/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts
+++ b/invokeai/frontend/web/src/features/controlLayers/store/canvasV2Slice.ts
@@ -132,6 +132,7 @@ const initialState: CanvasV2State = {
refinerStart: 0.8,
},
session: {
+ mode: 'generate',
isStaging: false,
stagedImages: [],
selectedStagedImageIndex: 0,
@@ -474,6 +475,7 @@ export const {
clipToBboxChanged,
canvasReset,
settingsDynamicGridToggled,
+ settingsAutoSaveToggled,
// All entities
entitySelected,
entityNameChanged,
@@ -601,6 +603,7 @@ export const {
sessionStagingAreaReset,
sessionNextStagedImageSelected,
sessionPrevStagedImageSelected,
+ sessionModeChanged,
} = canvasV2Slice.actions;
export const selectCanvasV2Slice = (state: RootState) => state.canvasV2;
diff --git a/invokeai/frontend/web/src/features/controlLayers/store/sessionReducers.ts b/invokeai/frontend/web/src/features/controlLayers/store/sessionReducers.ts
index 94c42aedea..8d2aeaeb11 100644
--- a/invokeai/frontend/web/src/features/controlLayers/store/sessionReducers.ts
+++ b/invokeai/frontend/web/src/features/controlLayers/store/sessionReducers.ts
@@ -1,5 +1,5 @@
import type { PayloadAction, SliceCaseReducers } from '@reduxjs/toolkit';
-import type { CanvasV2State, StagingAreaImage } from 'features/controlLayers/store/types';
+import type { CanvasV2State, SessionMode, StagingAreaImage } from 'features/controlLayers/store/types';
export const sessionReducers = {
sessionStartedStaging: (state) => {
@@ -45,4 +45,8 @@ export const sessionReducers = {
state.tool.selectedBuffer = null;
}
},
+ sessionModeChanged: (state, action: PayloadAction<{ mode: SessionMode }>) => {
+ const { mode } = action.payload;
+ state.session.mode = mode;
+ },
} satisfies SliceCaseReducers;
diff --git a/invokeai/frontend/web/src/features/controlLayers/store/settingsReducers.ts b/invokeai/frontend/web/src/features/controlLayers/store/settingsReducers.ts
index 5001c2f06a..324077f8eb 100644
--- a/invokeai/frontend/web/src/features/controlLayers/store/settingsReducers.ts
+++ b/invokeai/frontend/web/src/features/controlLayers/store/settingsReducers.ts
@@ -8,4 +8,7 @@ export const settingsReducers = {
settingsDynamicGridToggled: (state) => {
state.settings.dynamicGrid = !state.settings.dynamicGrid;
},
+ settingsAutoSaveToggled: (state) => {
+ state.settings.autoSave = !state.settings.autoSave;
+ },
} satisfies SliceCaseReducers;
diff --git a/invokeai/frontend/web/src/features/controlLayers/store/types.ts b/invokeai/frontend/web/src/features/controlLayers/store/types.ts
index 8b483d7899..d9b7d52a53 100644
--- a/invokeai/frontend/web/src/features/controlLayers/store/types.ts
+++ b/invokeai/frontend/web/src/features/controlLayers/store/types.ts
@@ -834,6 +834,8 @@ export type StagingAreaImage = {
offsetY: number;
};
+export type SessionMode = 'generate' | 'compose';
+
export type CanvasV2State = {
_version: 3;
selectedEntityIdentifier: CanvasEntityIdentifier | null;
@@ -929,6 +931,7 @@ export type CanvasV2State = {
refinerStart: number;
};
session: {
+ mode: SessionMode;
isStaging: boolean;
stagedImages: StagingAreaImage[];
selectedStagedImageIndex: number;
diff --git a/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSD1Graph.ts b/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSD1Graph.ts
index 47d5fe44a0..53f8875152 100644
--- a/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSD1Graph.ts
+++ b/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSD1Graph.ts
@@ -41,7 +41,7 @@ export const buildSD1Graph = async (state: RootState, manager: CanvasManager): P
const generationMode = manager.compositor.getGenerationMode();
log.debug({ generationMode }, 'Building SD1/SD2 graph');
- const { bbox, params } = state.canvasV2;
+ const { bbox, params, session, settings } = state.canvasV2;
const {
model,
@@ -249,10 +249,11 @@ export const buildSD1Graph = async (state: RootState, manager: CanvasManager): P
canvasOutput = addWatermarker(g, canvasOutput);
}
- // This is the terminal node and must always save to gallery.
+ const shouldSaveToGallery = session.mode === 'generate' || settings.autoSave;
+
g.updateNode(canvasOutput, {
id: CANVAS_OUTPUT,
- is_intermediate: false,
+ is_intermediate: !shouldSaveToGallery,
use_cache: false,
board: getBoardField(state),
});
diff --git a/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSDXLGraph.ts b/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSDXLGraph.ts
index ee34f09172..d55d632c6a 100644
--- a/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSDXLGraph.ts
+++ b/invokeai/frontend/web/src/features/nodes/util/graph/generation/buildSDXLGraph.ts
@@ -40,7 +40,7 @@ export const buildSDXLGraph = async (state: RootState, manager: CanvasManager):
const generationMode = manager.compositor.getGenerationMode();
log.debug({ generationMode }, 'Building SDXL graph');
- const { bbox, params } = state.canvasV2;
+ const { bbox, params, session, settings } = state.canvasV2;
const {
model,
@@ -246,10 +246,11 @@ export const buildSDXLGraph = async (state: RootState, manager: CanvasManager):
canvasOutput = addWatermarker(g, canvasOutput);
}
- // This is the terminal node and must always save to gallery.
+ const shouldSaveToGallery = session.mode === 'generate' || settings.autoSave;
+
g.updateNode(canvasOutput, {
id: CANVAS_OUTPUT,
- is_intermediate: false,
+ is_intermediate: !shouldSaveToGallery,
use_cache: false,
board: getBoardField(state),
});
diff --git a/invokeai/frontend/web/src/services/events/onInvocationComplete.ts b/invokeai/frontend/web/src/services/events/onInvocationComplete.ts
new file mode 100644
index 0000000000..b87713379e
--- /dev/null
+++ b/invokeai/frontend/web/src/services/events/onInvocationComplete.ts
@@ -0,0 +1,145 @@
+import { logger } from 'app/logging/logger';
+import type { AppDispatch, RootState } from 'app/store/store';
+import type { SerializableObject } from 'common/types';
+import { deepClone } from 'common/util/deepClone';
+import { sessionImageStaged } from 'features/controlLayers/store/canvasV2Slice';
+import { boardIdSelected, galleryViewChanged, imageSelected, offsetChanged } from 'features/gallery/store/gallerySlice';
+import { $nodeExecutionStates, upsertExecutionState } from 'features/nodes/hooks/useExecutionState';
+import { zNodeStatus } from 'features/nodes/types/invocation';
+import { boardsApi } from 'services/api/endpoints/boards';
+import { getImageDTO, imagesApi } from 'services/api/endpoints/images';
+import type { ImageDTO } from 'services/api/types';
+import { getCategories, getListImagesUrl } from 'services/api/util';
+import type { InvocationCompleteEvent, InvocationDenoiseProgressEvent } from 'services/events/types';
+
+const log = logger('events');
+
+export const buildOnInvocationComplete = (
+ getState: () => RootState,
+ dispatch: AppDispatch,
+ nodeTypeDenylist: string[],
+ setLastProgressEvent: (event: InvocationDenoiseProgressEvent | null) => void,
+ setLastCanvasProgressEvent: (event: InvocationDenoiseProgressEvent | null) => void
+) => {
+ const addImageToGallery = (imageDTO: ImageDTO) => {
+ if (imageDTO.is_intermediate) {
+ return;
+ }
+
+ // update the total images for the board
+ dispatch(
+ boardsApi.util.updateQueryData('getBoardImagesTotal', imageDTO.board_id ?? 'none', (draft) => {
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
+ draft.total += 1;
+ })
+ );
+
+ dispatch(
+ imagesApi.util.invalidateTags([
+ { type: 'Board', id: imageDTO.board_id ?? 'none' },
+ {
+ type: 'ImageList',
+ id: getListImagesUrl({
+ board_id: imageDTO.board_id ?? 'none',
+ categories: getCategories(imageDTO),
+ }),
+ },
+ ])
+ );
+
+ const { shouldAutoSwitch, galleryView, selectedBoardId } = getState().gallery;
+
+ // If auto-switch is enabled, select the new image
+ if (shouldAutoSwitch) {
+ // if auto-add is enabled, switch the gallery view and board if needed as the image comes in
+ if (galleryView !== 'images') {
+ dispatch(galleryViewChanged('images'));
+ }
+
+ if (imageDTO.board_id && imageDTO.board_id !== selectedBoardId) {
+ dispatch(
+ boardIdSelected({
+ boardId: imageDTO.board_id,
+ selectedImageName: imageDTO.image_name,
+ })
+ );
+ }
+
+ dispatch(offsetChanged({ offset: 0 }));
+
+ if (!imageDTO.board_id && selectedBoardId !== 'none') {
+ dispatch(
+ boardIdSelected({
+ boardId: 'none',
+ selectedImageName: imageDTO.image_name,
+ })
+ );
+ }
+
+ dispatch(imageSelected(imageDTO));
+ }
+ };
+
+ return async (data: InvocationCompleteEvent) => {
+ log.debug(
+ { data } as SerializableObject,
+ `Invocation complete (${data.invocation.type}, ${data.invocation_source_id})`
+ );
+
+ const { result, invocation_source_id } = data;
+
+ // Update the node execution states - the image output is handled below
+ if (data.origin === 'workflows') {
+ const nes = deepClone($nodeExecutionStates.get()[invocation_source_id]);
+ if (nes) {
+ nes.status = zNodeStatus.enum.COMPLETED;
+ if (nes.progress !== null) {
+ nes.progress = 1;
+ }
+ nes.outputs.push(result);
+ upsertExecutionState(nes.nodeId, nes);
+ }
+ }
+
+ // This complete event has an associated image output
+ if (
+ (data.result.type === 'image_output' || data.result.type === 'canvas_v2_mask_and_crop_output') &&
+ !nodeTypeDenylist.includes(data.invocation.type)
+ ) {
+ const { image_name } = data.result.image;
+ const { session } = getState().canvasV2;
+
+ const imageDTO = await getImageDTO(image_name);
+
+ if (!imageDTO) {
+ log.error({ data } as SerializableObject, 'Failed to fetch image DTO after generation');
+ return;
+ }
+
+ if (data.origin === 'canvas') {
+ if (data.invocation_source_id !== 'canvas_output') {
+ // Not a canvas output image - ignore
+ return;
+ }
+ if (session.mode === 'compose' && session.isStaging) {
+ if (data.result.type === 'canvas_v2_mask_and_crop_output') {
+ const { offset_x, offset_y } = data.result;
+ if (session.isStaging) {
+ dispatch(sessionImageStaged({ stagingAreaImage: { imageDTO, offsetX: offset_x, offsetY: offset_y } }));
+ }
+ } else if (data.result.type === 'image_output') {
+ if (session.isStaging) {
+ dispatch(sessionImageStaged({ stagingAreaImage: { imageDTO, offsetX: 0, offsetY: 0 } }));
+ }
+ }
+ addImageToGallery(imageDTO);
+ } else {
+ addImageToGallery(imageDTO);
+ setLastCanvasProgressEvent(null);
+ }
+ }
+ }
+
+ setLastProgressEvent(null);
+ };
+};
diff --git a/invokeai/frontend/web/src/services/events/setEventListeners.tsx b/invokeai/frontend/web/src/services/events/setEventListeners.tsx
index 3c08c7953e..9468dd707f 100644
--- a/invokeai/frontend/web/src/services/events/setEventListeners.tsx
+++ b/invokeai/frontend/web/src/services/events/setEventListeners.tsx
@@ -7,8 +7,6 @@ import { $queueId } from 'app/store/nanostores/queueId';
import type { AppDispatch, RootState } from 'app/store/store';
import type { SerializableObject } from 'common/types';
import { deepClone } from 'common/util/deepClone';
-import { sessionImageStaged } from 'features/controlLayers/store/canvasV2Slice';
-import { boardIdSelected, galleryViewChanged, imageSelected, offsetChanged } from 'features/gallery/store/gallerySlice';
import { $nodeExecutionStates, upsertExecutionState } from 'features/nodes/hooks/useExecutionState';
import { zNodeStatus } from 'features/nodes/types/invocation';
import ErrorToastDescription, { getTitleFromErrorType } from 'features/toast/ErrorToastDescription';
@@ -17,11 +15,9 @@ import { t } from 'i18next';
import { forEach } from 'lodash-es';
import { atom, computed } from 'nanostores';
import { api, LIST_TAG } from 'services/api';
-import { boardsApi } from 'services/api/endpoints/boards';
-import { imagesApi } from 'services/api/endpoints/images';
import { modelsApi } from 'services/api/endpoints/models';
import { queueApi, queueItemsAdapter } from 'services/api/endpoints/queue';
-import { getCategories, getListImagesUrl } from 'services/api/util';
+import { buildOnInvocationComplete } from 'services/events/onInvocationComplete';
import type { ClientToServerEvents, InvocationDenoiseProgressEvent, ServerToClientEvents } from 'services/events/types';
import type { Socket } from 'socket.io-client';
@@ -147,116 +143,14 @@ export const setEventListeners = ({ socket, dispatch, getState, setIsConnected }
}
});
- socket.on('invocation_complete', async (data) => {
- log.debug(
- { data } as SerializableObject,
- `Invocation complete (${data.invocation.type}, ${data.invocation_source_id})`
- );
-
- const { result, invocation_source_id } = data;
-
- if (data.origin === 'workflows') {
- const nes = deepClone($nodeExecutionStates.get()[invocation_source_id]);
- if (nes) {
- nes.status = zNodeStatus.enum.COMPLETED;
- if (nes.progress !== null) {
- nes.progress = 1;
- }
- nes.outputs.push(result);
- upsertExecutionState(nes.nodeId, nes);
- }
- }
-
- // This complete event has an associated image output
- if (
- (data.result.type === 'image_output' || data.result.type === 'canvas_v2_mask_and_crop_output') &&
- !nodeTypeDenylist.includes(data.invocation.type)
- ) {
- const { image_name } = data.result.image;
- const { gallery, canvasV2 } = getState();
-
- // This populates the `getImageDTO` cache
- const imageDTORequest = dispatch(
- imagesApi.endpoints.getImageDTO.initiate(image_name, {
- forceRefetch: true,
- })
- );
-
- const imageDTO = await imageDTORequest.unwrap();
- imageDTORequest.unsubscribe();
-
- // handle tab-specific logic
- if (data.origin === 'canvas' && data.invocation_source_id === 'canvas_output') {
- if (data.result.type === 'canvas_v2_mask_and_crop_output') {
- const { offset_x, offset_y } = data.result;
- if (canvasV2.session.isStaging) {
- dispatch(sessionImageStaged({ stagingAreaImage: { imageDTO, offsetX: offset_x, offsetY: offset_y } }));
- }
- } else if (data.result.type === 'image_output') {
- if (canvasV2.session.isStaging) {
- dispatch(sessionImageStaged({ stagingAreaImage: { imageDTO, offsetX: 0, offsetY: 0 } }));
- }
- }
- }
-
- if (!imageDTO.is_intermediate) {
- // update the total images for the board
- dispatch(
- boardsApi.util.updateQueryData('getBoardImagesTotal', imageDTO.board_id ?? 'none', (draft) => {
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
- draft.total += 1;
- })
- );
-
- dispatch(
- imagesApi.util.invalidateTags([
- { type: 'Board', id: imageDTO.board_id ?? 'none' },
- {
- type: 'ImageList',
- id: getListImagesUrl({
- board_id: imageDTO.board_id ?? 'none',
- categories: getCategories(imageDTO),
- }),
- },
- ])
- );
-
- const { shouldAutoSwitch } = gallery;
-
- // If auto-switch is enabled, select the new image
- if (shouldAutoSwitch) {
- // if auto-add is enabled, switch the gallery view and board if needed as the image comes in
- if (gallery.galleryView !== 'images') {
- dispatch(galleryViewChanged('images'));
- }
-
- if (imageDTO.board_id && imageDTO.board_id !== gallery.selectedBoardId) {
- dispatch(
- boardIdSelected({
- boardId: imageDTO.board_id,
- selectedImageName: imageDTO.image_name,
- })
- );
- }
-
- dispatch(offsetChanged({ offset: 0 }));
-
- if (!imageDTO.board_id && gallery.selectedBoardId !== 'none') {
- dispatch(
- boardIdSelected({
- boardId: 'none',
- selectedImageName: imageDTO.image_name,
- })
- );
- }
-
- dispatch(imageSelected(imageDTO));
- }
- }
- }
-
- $lastProgressEvent.set(null);
- });
+ const onInvocationComplete = buildOnInvocationComplete(
+ getState,
+ dispatch,
+ nodeTypeDenylist,
+ $lastProgressEvent.set,
+ $lastCanvasProgressEvent.set
+ );
+ socket.on('invocation_complete', onInvocationComplete);
socket.on('model_load_started', (data) => {
const { config, submodel_type } = data;