chore(ui): format

Lots of changed bc the line length is now 120. May as well do it now.
This commit is contained in:
psychedelicious 2024-01-27 20:55:55 +11:00
parent b922ee566a
commit 189c430e46
568 changed files with 3602 additions and 11172 deletions

View File

@ -1,13 +1,7 @@
{ {
"entry": ["src/main.tsx"], "entry": ["src/main.tsx"],
"extensions": [".ts", ".tsx"], "extensions": [".ts", ".tsx"],
"ignorePatterns": [ "ignorePatterns": ["**/node_modules/**", "dist/**", "public/**", "**/*.stories.tsx", "config/**"],
"**/node_modules/**",
"dist/**",
"public/**",
"**/*.stories.tsx",
"config/**"
],
"ignoreUnresolved": [], "ignoreUnresolved": [],
"ignoreUnimported": ["src/i18.d.ts", "vite.config.ts", "src/vite-env.d.ts"], "ignoreUnimported": ["src/i18.d.ts", "vite.config.ts", "src/vite-env.d.ts"],
"respectGitignore": true, "respectGitignore": true,

View File

@ -1463,9 +1463,7 @@
}, },
"compositingCoherencePass": { "compositingCoherencePass": {
"heading": "Coherence Pass", "heading": "Coherence Pass",
"paragraphs": [ "paragraphs": ["A second round of denoising helps to composite the Inpainted/Outpainted image."]
"A second round of denoising helps to composite the Inpainted/Outpainted image."
]
}, },
"compositingCoherenceMode": { "compositingCoherenceMode": {
"heading": "Mode", "heading": "Mode",
@ -1473,10 +1471,7 @@
}, },
"compositingCoherenceSteps": { "compositingCoherenceSteps": {
"heading": "Steps", "heading": "Steps",
"paragraphs": [ "paragraphs": ["Number of denoising steps used in the Coherence Pass.", "Same as the main Steps parameter."]
"Number of denoising steps used in the Coherence Pass.",
"Same as the main Steps parameter."
]
}, },
"compositingStrength": { "compositingStrength": {
"heading": "Strength", "heading": "Strength",
@ -1498,15 +1493,11 @@
}, },
"controlNetControlMode": { "controlNetControlMode": {
"heading": "Control Mode", "heading": "Control Mode",
"paragraphs": [ "paragraphs": ["Lends more weight to either the prompt or ControlNet."]
"Lends more weight to either the prompt or ControlNet."
]
}, },
"controlNetResizeMode": { "controlNetResizeMode": {
"heading": "Resize Mode", "heading": "Resize Mode",
"paragraphs": [ "paragraphs": ["How the ControlNet image will be fit to the image output size."]
"How the ControlNet image will be fit to the image output size."
]
}, },
"controlNet": { "controlNet": {
"heading": "ControlNet", "heading": "ControlNet",
@ -1516,9 +1507,7 @@
}, },
"controlNetWeight": { "controlNetWeight": {
"heading": "Weight", "heading": "Weight",
"paragraphs": [ "paragraphs": ["How strongly the ControlNet will impact the generated image."]
"How strongly the ControlNet will impact the generated image."
]
}, },
"dynamicPrompts": { "dynamicPrompts": {
"heading": "Dynamic Prompts", "heading": "Dynamic Prompts",
@ -1530,9 +1519,7 @@
}, },
"dynamicPromptsMaxPrompts": { "dynamicPromptsMaxPrompts": {
"heading": "Max Prompts", "heading": "Max Prompts",
"paragraphs": [ "paragraphs": ["Limits the number of prompts that can be generated by Dynamic Prompts."]
"Limits the number of prompts that can be generated by Dynamic Prompts."
]
}, },
"dynamicPromptsSeedBehaviour": { "dynamicPromptsSeedBehaviour": {
"heading": "Seed Behaviour", "heading": "Seed Behaviour",
@ -1549,9 +1536,7 @@
}, },
"lora": { "lora": {
"heading": "LoRA Weight", "heading": "LoRA Weight",
"paragraphs": [ "paragraphs": ["Higher LoRA weight will lead to larger impacts on the final image."]
"Higher LoRA weight will lead to larger impacts on the final image."
]
}, },
"noiseUseCPU": { "noiseUseCPU": {
"heading": "Use CPU Noise", "heading": "Use CPU Noise",
@ -1563,9 +1548,7 @@
}, },
"paramCFGScale": { "paramCFGScale": {
"heading": "CFG Scale", "heading": "CFG Scale",
"paragraphs": [ "paragraphs": ["Controls how much your prompt influences the generation process."]
"Controls how much your prompt influences the generation process."
]
}, },
"paramCFGRescaleMultiplier": { "paramCFGRescaleMultiplier": {
"heading": "CFG Rescale Multiplier", "heading": "CFG Rescale Multiplier",
@ -1617,9 +1600,7 @@
}, },
"paramVAE": { "paramVAE": {
"heading": "VAE", "heading": "VAE",
"paragraphs": [ "paragraphs": ["Model used for translating AI output into the final image."]
"Model used for translating AI output into the final image."
]
}, },
"paramVAEPrecision": { "paramVAEPrecision": {
"heading": "VAE Precision", "heading": "VAE Precision",

View File

@ -6,9 +6,7 @@ const OPENAPI_URL = 'http://127.0.0.1:9090/openapi.json';
const OUTPUT_FILE = 'src/services/api/schema.ts'; const OUTPUT_FILE = 'src/services/api/schema.ts';
async function main() { async function main() {
process.stdout.write( process.stdout.write(`Generating types "${OPENAPI_URL}" --> "${OUTPUT_FILE}"...`);
`Generating types "${OPENAPI_URL}" --> "${OUTPUT_FILE}"...`
);
const types = await openapiTS(OPENAPI_URL, { const types = await openapiTS(OPENAPI_URL, {
exportType: true, exportType: true,
transform: (schemaObject) => { transform: (schemaObject) => {

View File

@ -45,8 +45,7 @@ const App = ({ config = DEFAULT_CONFIG, selectedImage }: Props) => {
useGlobalModifiersInit(); useGlobalModifiersInit();
useGlobalHotkeys(); useGlobalHotkeys();
const { dropzone, isHandlingUpload, setIsHandlingUpload } = const { dropzone, isHandlingUpload, setIsHandlingUpload } = useFullscreenDropzone();
useFullscreenDropzone();
const handleReset = useCallback(() => { const handleReset = useCallback(() => {
clearStorage(); clearStorage();
@ -70,10 +69,7 @@ const App = ({ config = DEFAULT_CONFIG, selectedImage }: Props) => {
}, [dispatch]); }, [dispatch]);
return ( return (
<ErrorBoundary <ErrorBoundary onReset={handleReset} FallbackComponent={AppErrorBoundaryFallback}>
onReset={handleReset}
FallbackComponent={AppErrorBoundaryFallback}
>
<Box <Box
id="invoke-app-wrapper" id="invoke-app-wrapper"
w="100vw" w="100vw"
@ -86,10 +82,7 @@ const App = ({ config = DEFAULT_CONFIG, selectedImage }: Props) => {
<InvokeTabs /> <InvokeTabs />
<AnimatePresence> <AnimatePresence>
{dropzone.isDragActive && isHandlingUpload && ( {dropzone.isDragActive && isHandlingUpload && (
<ImageUploadOverlay <ImageUploadOverlay dropzone={dropzone} setIsHandlingUpload={setIsHandlingUpload} />
dropzone={dropzone}
setIsHandlingUpload={setIsHandlingUpload}
/>
)} )}
</AnimatePresence> </AnimatePresence>
</Box> </Box>

View File

@ -1,19 +1,8 @@
import { import { Button, Flex, Heading, Link, Text, useToast } from '@invoke-ai/ui-library';
Button,
Flex,
Heading,
Link,
Text,
useToast,
} from '@invoke-ai/ui-library';
import newGithubIssueUrl from 'new-github-issue-url'; import newGithubIssueUrl from 'new-github-issue-url';
import { memo, useCallback, useMemo } from 'react'; import { memo, useCallback, useMemo } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { import { PiArrowCounterClockwiseBold, PiArrowSquareOutBold, PiCopyBold } from 'react-icons/pi';
PiArrowCounterClockwiseBold,
PiArrowSquareOutBold,
PiCopyBold,
} from 'react-icons/pi';
import { serializeError } from 'serialize-error'; import { serializeError } from 'serialize-error';
type Props = { type Props = {
@ -44,22 +33,8 @@ const AppErrorBoundaryFallback = ({ error, resetErrorBoundary }: Props) => {
[error.message, error.name] [error.message, error.name]
); );
return ( return (
<Flex <Flex layerStyle="body" w="100vw" h="100vh" alignItems="center" justifyContent="center" p={4}>
layerStyle="body" <Flex layerStyle="first" flexDir="column" borderRadius="base" justifyContent="center" gap={8} p={16}>
w="100vw"
h="100vh"
alignItems="center"
justifyContent="center"
p={4}
>
<Flex
layerStyle="first"
flexDir="column"
borderRadius="base"
justifyContent="center"
gap={8}
p={16}
>
<Heading>{t('common.somethingWentWrong')}</Heading> <Heading>{t('common.somethingWentWrong')}</Heading>
<Flex <Flex
layerStyle="second" layerStyle="second"
@ -75,19 +50,14 @@ const AppErrorBoundaryFallback = ({ error, resetErrorBoundary }: Props) => {
</Text> </Text>
</Flex> </Flex>
<Flex gap={4}> <Flex gap={4}>
<Button <Button leftIcon={<PiArrowCounterClockwiseBold />} onClick={resetErrorBoundary}>
leftIcon={<PiArrowCounterClockwiseBold />}
onClick={resetErrorBoundary}
>
{t('accessibility.resetUI')} {t('accessibility.resetUI')}
</Button> </Button>
<Button leftIcon={<PiCopyBold />} onClick={handleCopy}> <Button leftIcon={<PiCopyBold />} onClick={handleCopy}>
{t('common.copyError')} {t('common.copyError')}
</Button> </Button>
<Link href={url} isExternal> <Link href={url} isExternal>
<Button leftIcon={<PiArrowSquareOutBold />}> <Button leftIcon={<PiArrowSquareOutBold />}>{t('accessibility.createIssue')}</Button>
{t('accessibility.createIssue')}
</Button>
</Link> </Link>
</Flex> </Flex>
</Flex> </Flex>

View File

@ -1,13 +1,7 @@
import '@fontsource-variable/inter'; import '@fontsource-variable/inter';
import 'overlayscrollbars/overlayscrollbars.css'; import 'overlayscrollbars/overlayscrollbars.css';
import { import { ChakraProvider, DarkMode, extendTheme, theme as _theme, TOAST_OPTIONS } from '@invoke-ai/ui-library';
ChakraProvider,
DarkMode,
extendTheme,
theme as _theme,
TOAST_OPTIONS,
} from '@invoke-ai/ui-library';
import type { ReactNode } from 'react'; import type { ReactNode } from 'react';
import { memo, useEffect, useMemo } from 'react'; import { memo, useEffect, useMemo } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';

View File

@ -36,10 +36,7 @@ const Toaster = () => {
*/ */
export const useAppToaster = () => { export const useAppToaster = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const toaster = useCallback( const toaster = useCallback((arg: MakeToastArg) => dispatch(addToast(makeToast(arg))), [dispatch]);
(arg: MakeToastArg) => dispatch(addToast(makeToast(arg))),
[dispatch]
);
return toaster; return toaster;
}; };

View File

@ -6,10 +6,7 @@ import { useAppDispatch } from 'app/store/storeHooks';
import type { MapStore } from 'nanostores'; import type { MapStore } from 'nanostores';
import { atom, map } from 'nanostores'; import { atom, map } from 'nanostores';
import { useEffect, useMemo } from 'react'; import { useEffect, useMemo } from 'react';
import type { import type { ClientToServerEvents, ServerToClientEvents } from 'services/events/types';
ClientToServerEvents,
ServerToClientEvents,
} from 'services/events/types';
import { setEventListeners } from 'services/events/util/setEventListeners'; import { setEventListeners } from 'services/events/util/setEventListeners';
import type { ManagerOptions, Socket, SocketOptions } from 'socket.io-client'; import type { ManagerOptions, Socket, SocketOptions } from 'socket.io-client';
import { io } from 'socket.io-client'; import { io } from 'socket.io-client';
@ -45,9 +42,7 @@ export const useSocketIO = () => {
const socketOptions = useMemo(() => { const socketOptions = useMemo(() => {
const options: Partial<ManagerOptions & SocketOptions> = { const options: Partial<ManagerOptions & SocketOptions> = {
timeout: 60000, timeout: 60000,
path: baseUrl path: baseUrl ? '/ws/socket.io' : `${window.location.pathname}ws/socket.io`,
? '/ws/socket.io'
: `${window.location.pathname}ws/socket.io`,
autoConnect: false, // achtung! removing this breaks the dynamic middleware autoConnect: false, // achtung! removing this breaks the dynamic middleware
forceNew: true, forceNew: true,
}; };
@ -66,10 +61,7 @@ export const useSocketIO = () => {
return; return;
} }
const socket: Socket<ServerToClientEvents, ClientToServerEvents> = io( const socket: Socket<ServerToClientEvents, ClientToServerEvents> = io(socketUrl, socketOptions);
socketUrl,
socketOptions
);
setEventListeners({ dispatch, socket }); setEventListeners({ dispatch, socket });
socket.connect(); socket.connect();

View File

@ -30,20 +30,11 @@ export type LoggerNamespace =
| 'queue' | 'queue'
| 'dnd'; | 'dnd';
export const logger = (namespace: LoggerNamespace) => export const logger = (namespace: LoggerNamespace) => $logger.get().child({ namespace });
$logger.get().child({ namespace });
export const zLogLevel = z.enum([ export const zLogLevel = z.enum(['trace', 'debug', 'info', 'warn', 'error', 'fatal']);
'trace',
'debug',
'info',
'warn',
'error',
'fatal',
]);
export type LogLevel = z.infer<typeof zLogLevel>; export type LogLevel = z.infer<typeof zLogLevel>;
export const isLogLevel = (v: unknown): v is LogLevel => export const isLogLevel = (v: unknown): v is LogLevel => zLogLevel.safeParse(v).success;
zLogLevel.safeParse(v).success;
// Translate human-readable log levels to numbers, used for log filtering // Translate human-readable log levels to numbers, used for log filtering
export const LOG_LEVEL_MAP: Record<LogLevel, number> = { export const LOG_LEVEL_MAP: Record<LogLevel, number> = {

View File

@ -17,10 +17,7 @@ export const useLogger = (namespace: LoggerNamespace) => {
localStorage.setItem('ROARR_LOG', 'true'); localStorage.setItem('ROARR_LOG', 'true');
// Use a filter to show only logs of the given level // Use a filter to show only logs of the given level
localStorage.setItem( localStorage.setItem('ROARR_FILTER', `context.logLevel:>=${LOG_LEVEL_MAP[consoleLogLevel]}`);
'ROARR_FILTER',
`context.logLevel:>=${LOG_LEVEL_MAP[consoleLogLevel]}`
);
} else { } else {
// Disable console log output // Disable console log output
localStorage.setItem('ROARR_LOG', 'false'); localStorage.setItem('ROARR_LOG', 'false');

View File

@ -1,8 +1,4 @@
import { import { createDraftSafeSelectorCreator, createSelectorCreator, lruMemoize } from '@reduxjs/toolkit';
createDraftSafeSelectorCreator,
createSelectorCreator,
lruMemoize,
} from '@reduxjs/toolkit';
import type { GetSelectorsOptions } from '@reduxjs/toolkit/dist/entities/state_selectors'; import type { GetSelectorsOptions } from '@reduxjs/toolkit/dist/entities/state_selectors';
import { isEqual } from 'lodash-es'; import { isEqual } from 'lodash-es';

View File

@ -1,19 +1,12 @@
import { StorageError } from 'app/store/enhancers/reduxRemember/errors'; import { StorageError } from 'app/store/enhancers/reduxRemember/errors';
import { $projectId } from 'app/store/nanostores/projectId'; import { $projectId } from 'app/store/nanostores/projectId';
import type { UseStore } from 'idb-keyval'; import type { UseStore } from 'idb-keyval';
import { import { clear, createStore as createIDBKeyValStore, get, set } from 'idb-keyval';
clear,
createStore as createIDBKeyValStore,
get,
set,
} from 'idb-keyval';
import { action, atom } from 'nanostores'; import { action, atom } from 'nanostores';
import type { Driver } from 'redux-remember'; import type { Driver } from 'redux-remember';
// Create a custom idb-keyval store (just needed to customize the name) // Create a custom idb-keyval store (just needed to customize the name)
export const $idbKeyValStore = atom<UseStore>( export const $idbKeyValStore = atom<UseStore>(createIDBKeyValStore('invoke', 'invoke-store'));
createIDBKeyValStore('invoke', 'invoke-store')
);
export const clearIdbKeyValStore = action($idbKeyValStore, 'clear', (store) => { export const clearIdbKeyValStore = action($idbKeyValStore, 'clear', (store) => {
clear(store.get()); clear(store.get());

View File

@ -4,13 +4,12 @@ import { diff } from 'jsondiffpatch';
/** /**
* Super simple logger middleware. Useful for debugging when the redux devtools are awkward. * Super simple logger middleware. Useful for debugging when the redux devtools are awkward.
*/ */
export const debugLoggerMiddleware: Middleware = export const debugLoggerMiddleware: Middleware = (api: MiddlewareAPI) => (next) => (action) => {
(api: MiddlewareAPI) => (next) => (action) => { const originalState = api.getState();
const originalState = api.getState(); console.log('REDUX: dispatching', action);
console.log('REDUX: dispatching', action); const result = next(action);
const result = next(action); const nextState = api.getState();
const nextState = api.getState(); console.log('REDUX: next state', nextState);
console.log('REDUX: next state', nextState); console.log('REDUX: diff', diff(originalState, nextState));
console.log('REDUX: diff', diff(originalState, nextState)); return result;
return result; };
};

View File

@ -35,8 +35,7 @@ export const actionSanitizer = <A extends UnknownAction>(action: A): A => {
if (socketGeneratorProgress.match(action)) { if (socketGeneratorProgress.match(action)) {
const sanitized = cloneDeep(action); const sanitized = cloneDeep(action);
if (sanitized.payload.data.progress_image) { if (sanitized.payload.data.progress_image) {
sanitized.payload.data.progress_image.dataURL = sanitized.payload.data.progress_image.dataURL = '<Progress image omitted>';
'<Progress image omitted>';
} }
return sanitized; return sanitized;
} }

View File

@ -1,9 +1,4 @@
import type { import type { ListenerEffect, TypedAddListener, TypedStartListening, UnknownAction } from '@reduxjs/toolkit';
ListenerEffect,
TypedAddListener,
TypedStartListening,
UnknownAction,
} from '@reduxjs/toolkit';
import { addListener, createListenerMiddleware } from '@reduxjs/toolkit'; import { addListener, createListenerMiddleware } from '@reduxjs/toolkit';
import { addGalleryImageClickedListener } from 'app/store/middleware/listenerMiddleware/listeners/galleryImageClicked'; import { addGalleryImageClickedListener } from 'app/store/middleware/listenerMiddleware/listeners/galleryImageClicked';
import type { AppDispatch, RootState } from 'app/store/store'; import type { AppDispatch, RootState } from 'app/store/store';
@ -47,10 +42,7 @@ import {
import { addImagesStarredListener } from './listeners/imagesStarred'; import { addImagesStarredListener } from './listeners/imagesStarred';
import { addImagesUnstarredListener } from './listeners/imagesUnstarred'; import { addImagesUnstarredListener } from './listeners/imagesUnstarred';
import { addImageToDeleteSelectedListener } from './listeners/imageToDeleteSelected'; import { addImageToDeleteSelectedListener } from './listeners/imageToDeleteSelected';
import { import { addImageUploadedFulfilledListener, addImageUploadedRejectedListener } from './listeners/imageUploaded';
addImageUploadedFulfilledListener,
addImageUploadedRejectedListener,
} from './listeners/imageUploaded';
import { addInitialImageSelectedListener } from './listeners/initialImageSelected'; import { addInitialImageSelectedListener } from './listeners/initialImageSelected';
import { addModelSelectedListener } from './listeners/modelSelected'; import { addModelSelectedListener } from './listeners/modelSelected';
import { addModelsLoadedListener } from './listeners/modelsLoaded'; import { addModelsLoadedListener } from './listeners/modelsLoaded';
@ -78,19 +70,11 @@ export const listenerMiddleware = createListenerMiddleware();
export type AppStartListening = TypedStartListening<RootState, AppDispatch>; export type AppStartListening = TypedStartListening<RootState, AppDispatch>;
export const startAppListening = export const startAppListening = listenerMiddleware.startListening as AppStartListening;
listenerMiddleware.startListening as AppStartListening;
export const addAppListener = addListener as TypedAddListener< export const addAppListener = addListener as TypedAddListener<RootState, AppDispatch>;
RootState,
AppDispatch
>;
export type AppListenerEffect = ListenerEffect< export type AppListenerEffect = ListenerEffect<UnknownAction, RootState, AppDispatch>;
UnknownAction,
RootState,
AppDispatch
>;
/** /**
* The RTK listener middleware is a lightweight alternative sagas/observables. * The RTK listener middleware is a lightweight alternative sagas/observables.

View File

@ -1,10 +1,6 @@
import { isAnyOf } from '@reduxjs/toolkit'; import { isAnyOf } from '@reduxjs/toolkit';
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { import { canvasBatchIdsReset, commitStagingAreaImage, discardStagedImages } from 'features/canvas/store/canvasSlice';
canvasBatchIdsReset,
commitStagingAreaImage,
discardStagedImages,
} from 'features/canvas/store/canvasSlice';
import { addToast } from 'features/system/store/systemSlice'; import { addToast } from 'features/system/store/systemSlice';
import { t } from 'i18next'; import { t } from 'i18next';
import { queueApi } from 'services/api/endpoints/queue'; import { queueApi } from 'services/api/endpoints/queue';
@ -23,10 +19,7 @@ export const addCommitStagingAreaImageListener = () => {
try { try {
const req = dispatch( const req = dispatch(
queueApi.endpoints.cancelByBatchIds.initiate( queueApi.endpoints.cancelByBatchIds.initiate({ batch_ids: batchIds }, { fixedCacheKey: 'cancelByBatchIds' })
{ batch_ids: batchIds },
{ fixedCacheKey: 'cancelByBatchIds' }
)
); );
const { canceled } = await req.unwrap(); const { canceled } = await req.unwrap();
req.reset(); req.reset();

View File

@ -12,15 +12,9 @@ export const appStarted = createAction('app/appStarted');
export const addFirstListImagesListener = () => { export const addFirstListImagesListener = () => {
startAppListening({ startAppListening({
matcher: imagesApi.endpoints.listImages.matchFulfilled, matcher: imagesApi.endpoints.listImages.matchFulfilled,
effect: async ( effect: async (action, { dispatch, unsubscribe, cancelActiveListeners }) => {
action,
{ dispatch, unsubscribe, cancelActiveListeners }
) => {
// Only run this listener on the first listImages request for no-board images // Only run this listener on the first listImages request for no-board images
if ( if (action.meta.arg.queryCacheKey !== getListImagesUrl({ board_id: 'none', categories: IMAGE_CATEGORIES })) {
action.meta.arg.queryCacheKey !==
getListImagesUrl({ board_id: 'none', categories: IMAGE_CATEGORIES })
) {
return; return;
} }

View File

@ -1,8 +1,5 @@
import { setInfillMethod } from 'features/parameters/store/generationSlice'; import { setInfillMethod } from 'features/parameters/store/generationSlice';
import { import { shouldUseNSFWCheckerChanged, shouldUseWatermarkerChanged } from 'features/system/store/systemSlice';
shouldUseNSFWCheckerChanged,
shouldUseWatermarkerChanged,
} from 'features/system/store/systemSlice';
import { appInfoApi } from 'services/api/endpoints/appInfo'; import { appInfoApi } from 'services/api/endpoints/appInfo';
import { startAppListening } from '..'; import { startAppListening } from '..';
@ -11,11 +8,7 @@ export const addAppConfigReceivedListener = () => {
startAppListening({ startAppListening({
matcher: appInfoApi.endpoints.getAppConfig.matchFulfilled, matcher: appInfoApi.endpoints.getAppConfig.matchFulfilled,
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
const { const { infill_methods = [], nsfw_methods = [], watermarking_methods = [] } = action.payload;
infill_methods = [],
nsfw_methods = [],
watermarking_methods = [],
} = action.payload;
const infillMethod = getState().generation.infillMethod; const infillMethod = getState().generation.infillMethod;
if (!infill_methods.includes(infillMethod)) { if (!infill_methods.includes(infillMethod)) {

View File

@ -1,8 +1,4 @@
import { import { createStandaloneToast, theme, TOAST_OPTIONS } from '@invoke-ai/ui-library';
createStandaloneToast,
theme,
TOAST_OPTIONS,
} from '@invoke-ai/ui-library';
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { parseify } from 'common/util/serialize'; import { parseify } from 'common/util/serialize';
import { zPydanticValidationError } from 'features/system/store/zodSchemas'; import { zPydanticValidationError } from 'features/system/store/zodSchemas';
@ -24,10 +20,7 @@ export const addBatchEnqueuedListener = () => {
effect: async (action) => { effect: async (action) => {
const response = action.payload; const response = action.payload;
const arg = action.meta.arg.originalArgs; const arg = action.meta.arg.originalArgs;
logger('queue').debug( logger('queue').debug({ enqueueResult: parseify(response) }, 'Batch enqueued');
{ enqueueResult: parseify(response) },
'Batch enqueued'
);
if (!toast.isActive('batch-queued')) { if (!toast.isActive('batch-queued')) {
toast({ toast({
@ -57,10 +50,7 @@ export const addBatchEnqueuedListener = () => {
status: 'error', status: 'error',
description: 'Unknown Error', description: 'Unknown Error',
}); });
logger('queue').error( logger('queue').error({ batchConfig: parseify(arg), error: parseify(response) }, t('queue.batchFailedToQueue'));
{ batchConfig: parseify(arg), error: parseify(response) },
t('queue.batchFailedToQueue')
);
return; return;
} }
@ -85,10 +75,7 @@ export const addBatchEnqueuedListener = () => {
status: 'error', status: 'error',
}); });
} }
logger('queue').error( logger('queue').error({ batchConfig: parseify(arg), error: parseify(response) }, t('queue.batchFailedToQueue'));
{ batchConfig: parseify(arg), error: parseify(response) },
t('queue.batchFailedToQueue')
);
}, },
}); });
}; };

View File

@ -22,13 +22,7 @@ export const addDeleteBoardAndImagesFulfilledListener = () => {
const { generation, canvas, nodes, controlAdapters } = getState(); const { generation, canvas, nodes, controlAdapters } = getState();
deleted_images.forEach((image_name) => { deleted_images.forEach((image_name) => {
const imageUsage = getImageUsage( const imageUsage = getImageUsage(generation, canvas, nodes, controlAdapters, image_name);
generation,
canvas,
nodes,
controlAdapters,
image_name
);
if (imageUsage.isInitialImage && !wasInitialImageReset) { if (imageUsage.isInitialImage && !wasInitialImageReset) {
dispatch(clearInitialImage()); dispatch(clearInitialImage());

View File

@ -1,13 +1,6 @@
import { isAnyOf } from '@reduxjs/toolkit'; import { isAnyOf } from '@reduxjs/toolkit';
import { import { boardIdSelected, galleryViewChanged, imageSelected } from 'features/gallery/store/gallerySlice';
boardIdSelected, import { ASSETS_CATEGORIES, IMAGE_CATEGORIES } from 'features/gallery/store/types';
galleryViewChanged,
imageSelected,
} from 'features/gallery/store/gallerySlice';
import {
ASSETS_CATEGORIES,
IMAGE_CATEGORIES,
} from 'features/gallery/store/types';
import { imagesApi } from 'services/api/endpoints/images'; import { imagesApi } from 'services/api/endpoints/images';
import { imagesSelectors } from 'services/api/util'; import { imagesSelectors } from 'services/api/util';
@ -16,53 +9,35 @@ import { startAppListening } from '..';
export const addBoardIdSelectedListener = () => { export const addBoardIdSelectedListener = () => {
startAppListening({ startAppListening({
matcher: isAnyOf(boardIdSelected, galleryViewChanged), matcher: isAnyOf(boardIdSelected, galleryViewChanged),
effect: async ( effect: async (action, { getState, dispatch, condition, cancelActiveListeners }) => {
action,
{ getState, dispatch, condition, cancelActiveListeners }
) => {
// Cancel any in-progress instances of this listener, we don't want to select an image from a previous board // Cancel any in-progress instances of this listener, we don't want to select an image from a previous board
cancelActiveListeners(); cancelActiveListeners();
const state = getState(); const state = getState();
const board_id = boardIdSelected.match(action) const board_id = boardIdSelected.match(action) ? action.payload.boardId : state.gallery.selectedBoardId;
? action.payload.boardId
: state.gallery.selectedBoardId;
const galleryView = galleryViewChanged.match(action) const galleryView = galleryViewChanged.match(action) ? action.payload : state.gallery.galleryView;
? action.payload
: state.gallery.galleryView;
// when a board is selected, we need to wait until the board has loaded *some* images, then select the first one // when a board is selected, we need to wait until the board has loaded *some* images, then select the first one
const categories = const categories = galleryView === 'images' ? IMAGE_CATEGORIES : ASSETS_CATEGORIES;
galleryView === 'images' ? IMAGE_CATEGORIES : ASSETS_CATEGORIES;
const queryArgs = { board_id: board_id ?? 'none', categories }; const queryArgs = { board_id: board_id ?? 'none', categories };
// wait until the board has some images - maybe it already has some from a previous fetch // wait until the board has some images - maybe it already has some from a previous fetch
// must use getState() to ensure we do not have stale state // must use getState() to ensure we do not have stale state
const isSuccess = await condition( const isSuccess = await condition(
() => () => imagesApi.endpoints.listImages.select(queryArgs)(getState()).isSuccess,
imagesApi.endpoints.listImages.select(queryArgs)(getState())
.isSuccess,
5000 5000
); );
if (isSuccess) { if (isSuccess) {
// the board was just changed - we can select the first image // the board was just changed - we can select the first image
const { data: boardImagesData } = const { data: boardImagesData } = imagesApi.endpoints.listImages.select(queryArgs)(getState());
imagesApi.endpoints.listImages.select(queryArgs)(getState());
if ( if (boardImagesData && boardIdSelected.match(action) && action.payload.selectedImageName) {
boardImagesData &&
boardIdSelected.match(action) &&
action.payload.selectedImageName
) {
const firstImage = imagesSelectors.selectAll(boardImagesData)[0]; const firstImage = imagesSelectors.selectAll(boardImagesData)[0];
const selectedImage = imagesSelectors.selectById( const selectedImage = imagesSelectors.selectById(boardImagesData, action.payload.selectedImageName);
boardImagesData,
action.payload.selectedImageName
);
dispatch(imageSelected(selectedImage || firstImage || null)); dispatch(imageSelected(selectedImage || firstImage || null));
} else { } else {

View File

@ -11,9 +11,7 @@ export const addCanvasCopiedToClipboardListener = () => {
startAppListening({ startAppListening({
actionCreator: canvasCopiedToClipboard, actionCreator: canvasCopiedToClipboard,
effect: async (action, { dispatch, getState }) => { effect: async (action, { dispatch, getState }) => {
const moduleLog = $logger const moduleLog = $logger.get().child({ namespace: 'canvasCopiedToClipboardListener' });
.get()
.child({ namespace: 'canvasCopiedToClipboardListener' });
const state = getState(); const state = getState();
try { try {

View File

@ -11,9 +11,7 @@ export const addCanvasDownloadedAsImageListener = () => {
startAppListening({ startAppListening({
actionCreator: canvasDownloadedAsImage, actionCreator: canvasDownloadedAsImage,
effect: async (action, { dispatch, getState }) => { effect: async (action, { dispatch, getState }) => {
const moduleLog = $logger const moduleLog = $logger.get().child({ namespace: 'canvasSavedToGalleryListener' });
.get()
.child({ namespace: 'canvasSavedToGalleryListener' });
const state = getState(); const state = getState();
let blob; let blob;
@ -32,9 +30,7 @@ export const addCanvasDownloadedAsImageListener = () => {
} }
downloadBlob(blob, 'canvas.png'); downloadBlob(blob, 'canvas.png');
dispatch( dispatch(addToast({ title: t('toast.canvasDownloaded'), status: 'success' }));
addToast({ title: t('toast.canvasDownloaded'), status: 'success' })
);
}, },
}); });
}; };

View File

@ -13,9 +13,7 @@ export const addCanvasMergedListener = () => {
startAppListening({ startAppListening({
actionCreator: canvasMerged, actionCreator: canvasMerged,
effect: async (action, { dispatch }) => { effect: async (action, { dispatch }) => {
const moduleLog = $logger const moduleLog = $logger.get().child({ namespace: 'canvasCopiedToClipboardListener' });
.get()
.child({ namespace: 'canvasCopiedToClipboardListener' });
const blob = await getFullBaseLayerBlob(); const blob = await getFullBaseLayerBlob();
if (!blob) { if (!blob) {

View File

@ -21,11 +21,7 @@ type AnyControlAdapterParamChangeAction =
| ReturnType<typeof controlAdapterProcessortTypeChanged> | ReturnType<typeof controlAdapterProcessortTypeChanged>
| ReturnType<typeof controlAdapterAutoConfigToggled>; | ReturnType<typeof controlAdapterAutoConfigToggled>;
const predicate: AnyListenerPredicate<RootState> = ( const predicate: AnyListenerPredicate<RootState> = (action, state, prevState) => {
action,
state,
prevState
) => {
const isActionMatched = const isActionMatched =
controlAdapterProcessorParamsChanged.match(action) || controlAdapterProcessorParamsChanged.match(action) ||
controlAdapterModelChanged.match(action) || controlAdapterModelChanged.match(action) ||
@ -40,12 +36,7 @@ const predicate: AnyListenerPredicate<RootState> = (
const { id } = action.payload; const { id } = action.payload;
const prevCA = selectControlAdapterById(prevState.controlAdapters, id); const prevCA = selectControlAdapterById(prevState.controlAdapters, id);
const ca = selectControlAdapterById(state.controlAdapters, id); const ca = selectControlAdapterById(state.controlAdapters, id);
if ( if (!prevCA || !isControlNetOrT2IAdapter(prevCA) || !ca || !isControlNetOrT2IAdapter(ca)) {
!prevCA ||
!isControlNetOrT2IAdapter(prevCA) ||
!ca ||
!isControlNetOrT2IAdapter(ca)
) {
return false; return false;
} }

View File

@ -64,37 +64,28 @@ export const addControlNetImageProcessedListener = () => {
); );
const enqueueResult = await req.unwrap(); const enqueueResult = await req.unwrap();
req.reset(); req.reset();
log.debug( log.debug({ enqueueResult: parseify(enqueueResult) }, t('queue.graphQueued'));
{ enqueueResult: parseify(enqueueResult) },
t('queue.graphQueued')
);
const [invocationCompleteAction] = await take( const [invocationCompleteAction] = await take(
(action): action is ReturnType<typeof socketInvocationComplete> => (action): action is ReturnType<typeof socketInvocationComplete> =>
socketInvocationComplete.match(action) && socketInvocationComplete.match(action) &&
action.payload.data.queue_batch_id === action.payload.data.queue_batch_id === enqueueResult.batch.batch_id &&
enqueueResult.batch.batch_id &&
action.payload.data.source_node_id === nodeId action.payload.data.source_node_id === nodeId
); );
// We still have to check the output type // We still have to check the output type
if (isImageOutput(invocationCompleteAction.payload.data.result)) { if (isImageOutput(invocationCompleteAction.payload.data.result)) {
const { image_name } = const { image_name } = invocationCompleteAction.payload.data.result.image;
invocationCompleteAction.payload.data.result.image;
// Wait for the ImageDTO to be received // Wait for the ImageDTO to be received
const [{ payload }] = await take( const [{ payload }] = await take(
(action) => (action) =>
imagesApi.endpoints.getImageDTO.matchFulfilled(action) && imagesApi.endpoints.getImageDTO.matchFulfilled(action) && action.payload.image_name === image_name
action.payload.image_name === image_name
); );
const processedControlImage = payload as ImageDTO; const processedControlImage = payload as ImageDTO;
log.debug( log.debug({ controlNetId: action.payload, processedControlImage }, 'ControlNet image processed');
{ controlNetId: action.payload, processedControlImage },
'ControlNet image processed'
);
// Update the processed image in the store // Update the processed image in the store
dispatch( dispatch(
@ -105,10 +96,7 @@ export const addControlNetImageProcessedListener = () => {
); );
} }
} catch (error) { } catch (error) {
log.error( log.error({ enqueueBatchArg: parseify(enqueueBatchArg) }, t('queue.graphFailedToQueue'));
{ enqueueBatchArg: parseify(enqueueBatchArg) },
t('queue.graphFailedToQueue')
);
if (error instanceof Object) { if (error instanceof Object) {
if ('data' in error && 'status' in error) { if ('data' in error && 'status' in error) {

View File

@ -2,10 +2,7 @@ import { logger } from 'app/logging/logger';
import { enqueueRequested } from 'app/store/actions'; import { enqueueRequested } from 'app/store/actions';
import openBase64ImageInTab from 'common/util/openBase64ImageInTab'; import openBase64ImageInTab from 'common/util/openBase64ImageInTab';
import { parseify } from 'common/util/serialize'; import { parseify } from 'common/util/serialize';
import { import { canvasBatchIdAdded, stagingAreaInitialized } from 'features/canvas/store/canvasSlice';
canvasBatchIdAdded,
stagingAreaInitialized,
} from 'features/canvas/store/canvasSlice';
import { blobToDataURL } from 'features/canvas/util/blobToDataURL'; import { blobToDataURL } from 'features/canvas/util/blobToDataURL';
import { getCanvasData } from 'features/canvas/util/getCanvasData'; import { getCanvasData } from 'features/canvas/util/getCanvasData';
import { getCanvasGenerationMode } from 'features/canvas/util/getCanvasGenerationMode'; import { getCanvasGenerationMode } from 'features/canvas/util/getCanvasGenerationMode';
@ -34,20 +31,14 @@ import { startAppListening } from '..';
export const addEnqueueRequestedCanvasListener = () => { export const addEnqueueRequestedCanvasListener = () => {
startAppListening({ startAppListening({
predicate: (action): action is ReturnType<typeof enqueueRequested> => predicate: (action): action is ReturnType<typeof enqueueRequested> =>
enqueueRequested.match(action) && enqueueRequested.match(action) && action.payload.tabName === 'unifiedCanvas',
action.payload.tabName === 'unifiedCanvas',
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
const log = logger('queue'); const log = logger('queue');
const { prepend } = action.payload; const { prepend } = action.payload;
const state = getState(); const state = getState();
const { const { layerState, boundingBoxCoordinates, boundingBoxDimensions, isMaskEnabled, shouldPreserveMaskedArea } =
layerState, state.canvas;
boundingBoxCoordinates,
boundingBoxDimensions,
isMaskEnabled,
shouldPreserveMaskedArea,
} = state.canvas;
// Build canvas blobs // Build canvas blobs
const canvasBlobsAndImageData = await getCanvasData( const canvasBlobsAndImageData = await getCanvasData(
@ -63,14 +54,10 @@ export const addEnqueueRequestedCanvasListener = () => {
return; return;
} }
const { baseBlob, baseImageData, maskBlob, maskImageData } = const { baseBlob, baseImageData, maskBlob, maskImageData } = canvasBlobsAndImageData;
canvasBlobsAndImageData;
// Determine the generation mode // Determine the generation mode
const generationMode = getCanvasGenerationMode( const generationMode = getCanvasGenerationMode(baseImageData, maskImageData);
baseImageData,
maskImageData
);
if (state.system.enableImageDebugging) { if (state.system.enableImageDebugging) {
const baseDataURL = await blobToDataURL(baseBlob); const baseDataURL = await blobToDataURL(baseBlob);
@ -115,12 +102,7 @@ export const addEnqueueRequestedCanvasListener = () => {
).unwrap(); ).unwrap();
} }
const graph = buildCanvasGraph( const graph = buildCanvasGraph(state, generationMode, canvasInitImage, canvasMaskImage);
state,
generationMode,
canvasInitImage,
canvasMaskImage
);
log.debug({ graph: parseify(graph) }, `Canvas graph built`); log.debug({ graph: parseify(graph) }, `Canvas graph built`);

View File

@ -11,9 +11,7 @@ import { startAppListening } from '..';
export const addEnqueueRequestedLinear = () => { export const addEnqueueRequestedLinear = () => {
startAppListening({ startAppListening({
predicate: (action): action is ReturnType<typeof enqueueRequested> => predicate: (action): action is ReturnType<typeof enqueueRequested> =>
enqueueRequested.match(action) && enqueueRequested.match(action) && (action.payload.tabName === 'txt2img' || action.payload.tabName === 'img2img'),
(action.payload.tabName === 'txt2img' ||
action.payload.tabName === 'img2img'),
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
const state = getState(); const state = getState();
const model = state.generation.model; const model = state.generation.model;

View File

@ -32,8 +32,7 @@ export const addGalleryImageClickedListener = () => {
const { imageDTO, shiftKey, ctrlKey, metaKey } = action.payload; const { imageDTO, shiftKey, ctrlKey, metaKey } = action.payload;
const state = getState(); const state = getState();
const queryArgs = selectListImagesQueryArgs(state); const queryArgs = selectListImagesQueryArgs(state);
const { data: listImagesData } = const { data: listImagesData } = imagesApi.endpoints.listImages.select(queryArgs)(state);
imagesApi.endpoints.listImages.select(queryArgs)(state);
if (!listImagesData) { if (!listImagesData) {
// Should never happen if we have clicked a gallery image // Should never happen if we have clicked a gallery image
@ -46,12 +45,8 @@ export const addGalleryImageClickedListener = () => {
if (shiftKey) { if (shiftKey) {
const rangeEndImageName = imageDTO.image_name; const rangeEndImageName = imageDTO.image_name;
const lastSelectedImage = selection[selection.length - 1]?.image_name; const lastSelectedImage = selection[selection.length - 1]?.image_name;
const lastClickedIndex = imageDTOs.findIndex( const lastClickedIndex = imageDTOs.findIndex((n) => n.image_name === lastSelectedImage);
(n) => n.image_name === lastSelectedImage const currentClickedIndex = imageDTOs.findIndex((n) => n.image_name === rangeEndImageName);
);
const currentClickedIndex = imageDTOs.findIndex(
(n) => n.image_name === rangeEndImageName
);
if (lastClickedIndex > -1 && currentClickedIndex > -1) { if (lastClickedIndex > -1 && currentClickedIndex > -1) {
// We have a valid range! // We have a valid range!
const start = Math.min(lastClickedIndex, currentClickedIndex); const start = Math.min(lastClickedIndex, currentClickedIndex);
@ -60,15 +55,8 @@ export const addGalleryImageClickedListener = () => {
dispatch(selectionChanged(selection.concat(imagesToSelect))); dispatch(selectionChanged(selection.concat(imagesToSelect)));
} }
} else if (ctrlKey || metaKey) { } else if (ctrlKey || metaKey) {
if ( if (selection.some((i) => i.image_name === imageDTO.image_name) && selection.length > 1) {
selection.some((i) => i.image_name === imageDTO.image_name) && dispatch(selectionChanged(selection.filter((n) => n.image_name !== imageDTO.image_name)));
selection.length > 1
) {
dispatch(
selectionChanged(
selection.filter((n) => n.image_name !== imageDTO.image_name)
)
);
} else { } else {
dispatch(selectionChanged(selection.concat(imageDTO))); dispatch(selectionChanged(selection.concat(imageDTO)));
} }

View File

@ -43,31 +43,21 @@ export const addRequestedSingleImageDeletionListener = () => {
dispatch(isModalOpenChanged(false)); dispatch(isModalOpenChanged(false));
const state = getState(); const state = getState();
const lastSelectedImage = const lastSelectedImage = state.gallery.selection[state.gallery.selection.length - 1]?.image_name;
state.gallery.selection[state.gallery.selection.length - 1]?.image_name;
if (imageDTO && imageDTO?.image_name === lastSelectedImage) { if (imageDTO && imageDTO?.image_name === lastSelectedImage) {
const { image_name } = imageDTO; const { image_name } = imageDTO;
const baseQueryArgs = selectListImagesQueryArgs(state); const baseQueryArgs = selectListImagesQueryArgs(state);
const { data } = const { data } = imagesApi.endpoints.listImages.select(baseQueryArgs)(state);
imagesApi.endpoints.listImages.select(baseQueryArgs)(state);
const cachedImageDTOs = data ? imagesSelectors.selectAll(data) : []; const cachedImageDTOs = data ? imagesSelectors.selectAll(data) : [];
const deletedImageIndex = cachedImageDTOs.findIndex( const deletedImageIndex = cachedImageDTOs.findIndex((i) => i.image_name === image_name);
(i) => i.image_name === image_name
);
const filteredImageDTOs = cachedImageDTOs.filter( const filteredImageDTOs = cachedImageDTOs.filter((i) => i.image_name !== image_name);
(i) => i.image_name !== image_name
);
const newSelectedImageIndex = clamp( const newSelectedImageIndex = clamp(deletedImageIndex, 0, filteredImageDTOs.length - 1);
deletedImageIndex,
0,
filteredImageDTOs.length - 1
);
const newSelectedImageDTO = filteredImageDTOs[newSelectedImageIndex]; const newSelectedImageDTO = filteredImageDTOs[newSelectedImageIndex];
@ -85,9 +75,7 @@ export const addRequestedSingleImageDeletionListener = () => {
imageDTOs.forEach((imageDTO) => { imageDTOs.forEach((imageDTO) => {
// reset init image if we deleted it // reset init image if we deleted it
if ( if (getState().generation.initialImage?.imageName === imageDTO.image_name) {
getState().generation.initialImage?.imageName === imageDTO.image_name
) {
dispatch(clearInitialImage()); dispatch(clearInitialImage());
} }
@ -95,8 +83,7 @@ export const addRequestedSingleImageDeletionListener = () => {
forEach(selectControlAdapterAll(getState().controlAdapters), (ca) => { forEach(selectControlAdapterAll(getState().controlAdapters), (ca) => {
if ( if (
ca.controlImage === imageDTO.image_name || ca.controlImage === imageDTO.image_name ||
(isControlNetOrT2IAdapter(ca) && (isControlNetOrT2IAdapter(ca) && ca.processedControlImage === imageDTO.image_name)
ca.processedControlImage === imageDTO.image_name)
) { ) {
dispatch( dispatch(
controlAdapterImageChanged({ controlAdapterImageChanged({
@ -120,10 +107,7 @@ export const addRequestedSingleImageDeletionListener = () => {
} }
forEach(node.data.inputs, (input) => { forEach(node.data.inputs, (input) => {
if ( if (isImageFieldInputInstance(input) && input.value?.image_name === imageDTO.image_name) {
isImageFieldInputInstance(input) &&
input.value?.image_name === imageDTO.image_name
) {
dispatch( dispatch(
fieldImageValueChanged({ fieldImageValueChanged({
nodeId: node.data.id, nodeId: node.data.id,
@ -137,24 +121,16 @@ export const addRequestedSingleImageDeletionListener = () => {
}); });
// Delete from server // Delete from server
const { requestId } = dispatch( const { requestId } = dispatch(imagesApi.endpoints.deleteImage.initiate(imageDTO));
imagesApi.endpoints.deleteImage.initiate(imageDTO)
);
// Wait for successful deletion, then trigger boards to re-fetch // Wait for successful deletion, then trigger boards to re-fetch
const wasImageDeleted = await condition( const wasImageDeleted = await condition(
(action) => (action) => imagesApi.endpoints.deleteImage.matchFulfilled(action) && action.meta.requestId === requestId,
imagesApi.endpoints.deleteImage.matchFulfilled(action) &&
action.meta.requestId === requestId,
30000 30000
); );
if (wasImageDeleted) { if (wasImageDeleted) {
dispatch( dispatch(api.util.invalidateTags([{ type: 'Board', id: imageDTO.board_id ?? 'none' }]));
api.util.invalidateTags([
{ type: 'Board', id: imageDTO.board_id ?? 'none' },
])
);
} }
}, },
}); });
@ -176,17 +152,12 @@ export const addRequestedMultipleImageDeletionListener = () => {
try { try {
// Delete from server // Delete from server
await dispatch( await dispatch(imagesApi.endpoints.deleteImages.initiate({ imageDTOs })).unwrap();
imagesApi.endpoints.deleteImages.initiate({ imageDTOs })
).unwrap();
const state = getState(); const state = getState();
const queryArgs = selectListImagesQueryArgs(state); const queryArgs = selectListImagesQueryArgs(state);
const { data } = const { data } = imagesApi.endpoints.listImages.select(queryArgs)(state);
imagesApi.endpoints.listImages.select(queryArgs)(state);
const newSelectedImageDTO = data const newSelectedImageDTO = data ? imagesSelectors.selectAll(data)[0] : undefined;
? imagesSelectors.selectAll(data)[0]
: undefined;
if (newSelectedImageDTO) { if (newSelectedImageDTO) {
dispatch(imageSelected(newSelectedImageDTO)); dispatch(imageSelected(newSelectedImageDTO));
@ -204,10 +175,7 @@ export const addRequestedMultipleImageDeletionListener = () => {
imageDTOs.forEach((imageDTO) => { imageDTOs.forEach((imageDTO) => {
// reset init image if we deleted it // reset init image if we deleted it
if ( if (getState().generation.initialImage?.imageName === imageDTO.image_name) {
getState().generation.initialImage?.imageName ===
imageDTO.image_name
) {
dispatch(clearInitialImage()); dispatch(clearInitialImage());
} }
@ -215,8 +183,7 @@ export const addRequestedMultipleImageDeletionListener = () => {
forEach(selectControlAdapterAll(getState().controlAdapters), (ca) => { forEach(selectControlAdapterAll(getState().controlAdapters), (ca) => {
if ( if (
ca.controlImage === imageDTO.image_name || ca.controlImage === imageDTO.image_name ||
(isControlNetOrT2IAdapter(ca) && (isControlNetOrT2IAdapter(ca) && ca.processedControlImage === imageDTO.image_name)
ca.processedControlImage === imageDTO.image_name)
) { ) {
dispatch( dispatch(
controlAdapterImageChanged({ controlAdapterImageChanged({
@ -240,10 +207,7 @@ export const addRequestedMultipleImageDeletionListener = () => {
} }
forEach(node.data.inputs, (input) => { forEach(node.data.inputs, (input) => {
if ( if (isImageFieldInputInstance(input) && input.value?.image_name === imageDTO.image_name) {
isImageFieldInputInstance(input) &&
input.value?.image_name === imageDTO.image_name
) {
dispatch( dispatch(
fieldImageValueChanged({ fieldImageValueChanged({
nodeId: node.data.id, nodeId: node.data.id,
@ -295,10 +259,7 @@ export const addImageDeletedRejectedListener = () => {
matcher: imagesApi.endpoints.deleteImage.matchRejected, matcher: imagesApi.endpoints.deleteImage.matchRejected,
effect: (action) => { effect: (action) => {
const log = logger('images'); const log = logger('images');
log.debug( log.debug({ imageDTO: action.meta.arg.originalArgs }, 'Unable to delete image');
{ imageDTO: action.meta.arg.originalArgs },
'Unable to delete image'
);
}, },
}); });
}; };

View File

@ -6,16 +6,10 @@ import {
controlAdapterImageChanged, controlAdapterImageChanged,
controlAdapterIsEnabledChanged, controlAdapterIsEnabledChanged,
} from 'features/controlAdapters/store/controlAdaptersSlice'; } from 'features/controlAdapters/store/controlAdaptersSlice';
import type { import type { TypesafeDraggableData, TypesafeDroppableData } from 'features/dnd/types';
TypesafeDraggableData,
TypesafeDroppableData,
} from 'features/dnd/types';
import { imageSelected } from 'features/gallery/store/gallerySlice'; import { imageSelected } from 'features/gallery/store/gallerySlice';
import { fieldImageValueChanged } from 'features/nodes/store/nodesSlice'; import { fieldImageValueChanged } from 'features/nodes/store/nodesSlice';
import { import { initialImageChanged, selectOptimalDimension } from 'features/parameters/store/generationSlice';
initialImageChanged,
selectOptimalDimension,
} from 'features/parameters/store/generationSlice';
import { imagesApi } from 'services/api/endpoints/images'; import { imagesApi } from 'services/api/endpoints/images';
import { startAppListening } from '../'; import { startAppListening } from '../';
@ -35,15 +29,9 @@ export const addImageDroppedListener = () => {
if (activeData.payloadType === 'IMAGE_DTO') { if (activeData.payloadType === 'IMAGE_DTO') {
log.debug({ activeData, overData }, 'Image dropped'); log.debug({ activeData, overData }, 'Image dropped');
} else if (activeData.payloadType === 'GALLERY_SELECTION') { } else if (activeData.payloadType === 'GALLERY_SELECTION') {
log.debug( log.debug({ activeData, overData }, `Images (${getState().gallery.selection.length}) dropped`);
{ activeData, overData },
`Images (${getState().gallery.selection.length}) dropped`
);
} else if (activeData.payloadType === 'NODE_FIELD') { } else if (activeData.payloadType === 'NODE_FIELD') {
log.debug( log.debug({ activeData: parseify(activeData), overData: parseify(overData) }, 'Node field dropped');
{ activeData: parseify(activeData), overData: parseify(overData) },
'Node field dropped'
);
} else { } else {
log.debug({ activeData, overData }, `Unknown payload dropped`); log.debug({ activeData, overData }, `Unknown payload dropped`);
} }
@ -104,12 +92,7 @@ export const addImageDroppedListener = () => {
activeData.payloadType === 'IMAGE_DTO' && activeData.payloadType === 'IMAGE_DTO' &&
activeData.payload.imageDTO activeData.payload.imageDTO
) { ) {
dispatch( dispatch(setInitialCanvasImage(activeData.payload.imageDTO, selectOptimalDimension(getState())));
setInitialCanvasImage(
activeData.payload.imageDTO,
selectOptimalDimension(getState())
)
);
return; return;
} }
@ -191,10 +174,7 @@ export const addImageDroppedListener = () => {
/** /**
* Multiple images dropped on user board * Multiple images dropped on user board
*/ */
if ( if (overData.actionType === 'ADD_TO_BOARD' && activeData.payloadType === 'GALLERY_SELECTION') {
overData.actionType === 'ADD_TO_BOARD' &&
activeData.payloadType === 'GALLERY_SELECTION'
) {
const imageDTOs = getState().gallery.selection; const imageDTOs = getState().gallery.selection;
const { boardId } = overData.context; const { boardId } = overData.context;
dispatch( dispatch(
@ -209,10 +189,7 @@ export const addImageDroppedListener = () => {
/** /**
* Multiple images dropped on 'none' board * Multiple images dropped on 'none' board
*/ */
if ( if (overData.actionType === 'REMOVE_FROM_BOARD' && activeData.payloadType === 'GALLERY_SELECTION') {
overData.actionType === 'REMOVE_FROM_BOARD' &&
activeData.payloadType === 'GALLERY_SELECTION'
) {
const imageDTOs = getState().gallery.selection; const imageDTOs = getState().gallery.selection;
dispatch( dispatch(
imagesApi.endpoints.removeImagesFromBoard.initiate({ imagesApi.endpoints.removeImagesFromBoard.initiate({

View File

@ -1,9 +1,6 @@
import { imageDeletionConfirmed } from 'features/deleteImageModal/store/actions'; import { imageDeletionConfirmed } from 'features/deleteImageModal/store/actions';
import { selectImageUsage } from 'features/deleteImageModal/store/selectors'; import { selectImageUsage } from 'features/deleteImageModal/store/selectors';
import { import { imagesToDeleteSelected, isModalOpenChanged } from 'features/deleteImageModal/store/slice';
imagesToDeleteSelected,
isModalOpenChanged,
} from 'features/deleteImageModal/store/slice';
import { startAppListening } from '..'; import { startAppListening } from '..';

View File

@ -6,10 +6,7 @@ import {
controlAdapterIsEnabledChanged, controlAdapterIsEnabledChanged,
} from 'features/controlAdapters/store/controlAdaptersSlice'; } from 'features/controlAdapters/store/controlAdaptersSlice';
import { fieldImageValueChanged } from 'features/nodes/store/nodesSlice'; import { fieldImageValueChanged } from 'features/nodes/store/nodesSlice';
import { import { initialImageChanged, selectOptimalDimension } from 'features/parameters/store/generationSlice';
initialImageChanged,
selectOptimalDimension,
} from 'features/parameters/store/generationSlice';
import { addToast } from 'features/system/store/systemSlice'; import { addToast } from 'features/system/store/systemSlice';
import { t } from 'i18next'; import { t } from 'i18next';
import { omit } from 'lodash-es'; import { omit } from 'lodash-es';
@ -79,9 +76,7 @@ export const addImageUploadedFulfilledListener = () => {
} }
if (postUploadAction?.type === 'SET_CANVAS_INITIAL_IMAGE') { if (postUploadAction?.type === 'SET_CANVAS_INITIAL_IMAGE') {
dispatch( dispatch(setInitialCanvasImage(imageDTO, selectOptimalDimension(state)));
setInitialCanvasImage(imageDTO, selectOptimalDimension(state))
);
dispatch( dispatch(
addToast({ addToast({
...DEFAULT_UPLOADED_TOAST, ...DEFAULT_UPLOADED_TOAST,
@ -127,9 +122,7 @@ export const addImageUploadedFulfilledListener = () => {
if (postUploadAction?.type === 'SET_NODES_IMAGE') { if (postUploadAction?.type === 'SET_NODES_IMAGE') {
const { nodeId, fieldName } = postUploadAction; const { nodeId, fieldName } = postUploadAction;
dispatch( dispatch(fieldImageValueChanged({ nodeId, fieldName, value: imageDTO }));
fieldImageValueChanged({ nodeId, fieldName, value: imageDTO })
);
dispatch( dispatch(
addToast({ addToast({
...DEFAULT_UPLOADED_TOAST, ...DEFAULT_UPLOADED_TOAST,

View File

@ -11,11 +11,7 @@ export const addInitialImageSelectedListener = () => {
actionCreator: initialImageSelected, actionCreator: initialImageSelected,
effect: (action, { dispatch }) => { effect: (action, { dispatch }) => {
if (!action.payload) { if (!action.payload) {
dispatch( dispatch(addToast(makeToast({ title: t('toast.imageNotLoadedDesc'), status: 'error' })));
addToast(
makeToast({ title: t('toast.imageNotLoadedDesc'), status: 'error' })
)
);
return; return;
} }

View File

@ -5,10 +5,7 @@ import {
} from 'features/controlAdapters/store/controlAdaptersSlice'; } from 'features/controlAdapters/store/controlAdaptersSlice';
import { loraRemoved } from 'features/lora/store/loraSlice'; import { loraRemoved } from 'features/lora/store/loraSlice';
import { modelSelected } from 'features/parameters/store/actions'; import { modelSelected } from 'features/parameters/store/actions';
import { import { modelChanged, vaeSelected } from 'features/parameters/store/generationSlice';
modelChanged,
vaeSelected,
} from 'features/parameters/store/generationSlice';
import { zParameterModel } from 'features/parameters/types/parameterSchemas'; import { zParameterModel } from 'features/parameters/types/parameterSchemas';
import { addToast } from 'features/system/store/systemSlice'; import { addToast } from 'features/system/store/systemSlice';
import { makeToast } from 'features/system/util/makeToast'; import { makeToast } from 'features/system/util/makeToast';
@ -27,18 +24,14 @@ export const addModelSelectedListener = () => {
const result = zParameterModel.safeParse(action.payload); const result = zParameterModel.safeParse(action.payload);
if (!result.success) { if (!result.success) {
log.error( log.error({ error: result.error.format() }, 'Failed to parse main model');
{ error: result.error.format() },
'Failed to parse main model'
);
return; return;
} }
const newModel = result.data; const newModel = result.data;
const newBaseModel = newModel.base_model; const newBaseModel = newModel.base_model;
const didBaseModelChange = const didBaseModelChange = state.generation.model?.base_model !== newBaseModel;
state.generation.model?.base_model !== newBaseModel;
if (didBaseModelChange) { if (didBaseModelChange) {
// we may need to reset some incompatible submodels // we may need to reset some incompatible submodels
@ -62,9 +55,7 @@ export const addModelSelectedListener = () => {
// handle incompatible controlnets // handle incompatible controlnets
selectControlAdapterAll(state.controlAdapters).forEach((ca) => { selectControlAdapterAll(state.controlAdapters).forEach((ca) => {
if (ca.model?.base_model !== newBaseModel) { if (ca.model?.base_model !== newBaseModel) {
dispatch( dispatch(controlAdapterIsEnabledChanged({ id: ca.id, isEnabled: false }));
controlAdapterIsEnabledChanged({ id: ca.id, isEnabled: false })
);
modelsCleared += 1; modelsCleared += 1;
} }
}); });

View File

@ -6,41 +6,24 @@ import {
selectAllT2IAdapters, selectAllT2IAdapters,
} from 'features/controlAdapters/store/controlAdaptersSlice'; } from 'features/controlAdapters/store/controlAdaptersSlice';
import { loraRemoved } from 'features/lora/store/loraSlice'; import { loraRemoved } from 'features/lora/store/loraSlice';
import { import { modelChanged, vaeSelected } from 'features/parameters/store/generationSlice';
modelChanged, import { zParameterModel, zParameterVAEModel } from 'features/parameters/types/parameterSchemas';
vaeSelected,
} from 'features/parameters/store/generationSlice';
import {
zParameterModel,
zParameterVAEModel,
} from 'features/parameters/types/parameterSchemas';
import { refinerModelChanged } from 'features/sdxl/store/sdxlSlice'; import { refinerModelChanged } from 'features/sdxl/store/sdxlSlice';
import { forEach, some } from 'lodash-es'; import { forEach, some } from 'lodash-es';
import { import { mainModelsAdapterSelectors, modelsApi, vaeModelsAdapterSelectors } from 'services/api/endpoints/models';
mainModelsAdapterSelectors,
modelsApi,
vaeModelsAdapterSelectors,
} from 'services/api/endpoints/models';
import type { TypeGuardFor } from 'services/api/types'; import type { TypeGuardFor } from 'services/api/types';
import { startAppListening } from '..'; import { startAppListening } from '..';
export const addModelsLoadedListener = () => { export const addModelsLoadedListener = () => {
startAppListening({ startAppListening({
predicate: ( predicate: (action): action is TypeGuardFor<typeof modelsApi.endpoints.getMainModels.matchFulfilled> =>
action
): action is TypeGuardFor<
typeof modelsApi.endpoints.getMainModels.matchFulfilled
> =>
modelsApi.endpoints.getMainModels.matchFulfilled(action) && modelsApi.endpoints.getMainModels.matchFulfilled(action) &&
!action.meta.arg.originalArgs.includes('sdxl-refiner'), !action.meta.arg.originalArgs.includes('sdxl-refiner'),
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
// models loaded, we need to ensure the selected model is available and if not, select the first one // models loaded, we need to ensure the selected model is available and if not, select the first one
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `Main models loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`Main models loaded (${action.payload.ids.length})`
);
const currentModel = getState().generation.model; const currentModel = getState().generation.model;
const models = mainModelsAdapterSelectors.selectAll(action.payload); const models = mainModelsAdapterSelectors.selectAll(action.payload);
@ -67,10 +50,7 @@ export const addModelsLoadedListener = () => {
const result = zParameterModel.safeParse(models[0]); const result = zParameterModel.safeParse(models[0]);
if (!result.success) { if (!result.success) {
log.error( log.error({ error: result.error.format() }, 'Failed to parse main model');
{ error: result.error.format() },
'Failed to parse main model'
);
return; return;
} }
@ -78,20 +58,12 @@ export const addModelsLoadedListener = () => {
}, },
}); });
startAppListening({ startAppListening({
predicate: ( predicate: (action): action is TypeGuardFor<typeof modelsApi.endpoints.getMainModels.matchFulfilled> =>
action modelsApi.endpoints.getMainModels.matchFulfilled(action) && action.meta.arg.originalArgs.includes('sdxl-refiner'),
): action is TypeGuardFor<
typeof modelsApi.endpoints.getMainModels.matchFulfilled
> =>
modelsApi.endpoints.getMainModels.matchFulfilled(action) &&
action.meta.arg.originalArgs.includes('sdxl-refiner'),
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
// models loaded, we need to ensure the selected model is available and if not, select the first one // models loaded, we need to ensure the selected model is available and if not, select the first one
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `SDXL Refiner models loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`SDXL Refiner models loaded (${action.payload.ids.length})`
);
const currentModel = getState().sdxl.refinerModel; const currentModel = getState().sdxl.refinerModel;
const models = mainModelsAdapterSelectors.selectAll(action.payload); const models = mainModelsAdapterSelectors.selectAll(action.payload);
@ -122,10 +94,7 @@ export const addModelsLoadedListener = () => {
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
// VAEs loaded, need to reset the VAE is it's no longer available // VAEs loaded, need to reset the VAE is it's no longer available
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `VAEs loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`VAEs loaded (${action.payload.ids.length})`
);
const currentVae = getState().generation.vae; const currentVae = getState().generation.vae;
@ -136,9 +105,7 @@ export const addModelsLoadedListener = () => {
const isCurrentVAEAvailable = some( const isCurrentVAEAvailable = some(
action.payload.entities, action.payload.entities,
(m) => (m) => m?.model_name === currentVae?.model_name && m?.base_model === currentVae?.base_model
m?.model_name === currentVae?.model_name &&
m?.base_model === currentVae?.base_model
); );
if (isCurrentVAEAvailable) { if (isCurrentVAEAvailable) {
@ -156,10 +123,7 @@ export const addModelsLoadedListener = () => {
const result = zParameterVAEModel.safeParse(firstModel); const result = zParameterVAEModel.safeParse(firstModel);
if (!result.success) { if (!result.success) {
log.error( log.error({ error: result.error.format() }, 'Failed to parse VAE model');
{ error: result.error.format() },
'Failed to parse VAE model'
);
return; return;
} }
@ -171,19 +135,14 @@ export const addModelsLoadedListener = () => {
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
// LoRA models loaded - need to remove missing LoRAs from state // LoRA models loaded - need to remove missing LoRAs from state
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `LoRAs loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`LoRAs loaded (${action.payload.ids.length})`
);
const loras = getState().lora.loras; const loras = getState().lora.loras;
forEach(loras, (lora, id) => { forEach(loras, (lora, id) => {
const isLoRAAvailable = some( const isLoRAAvailable = some(
action.payload.entities, action.payload.entities,
(m) => (m) => m?.model_name === lora?.model_name && m?.base_model === lora?.base_model
m?.model_name === lora?.model_name &&
m?.base_model === lora?.base_model
); );
if (isLoRAAvailable) { if (isLoRAAvailable) {
@ -199,17 +158,12 @@ export const addModelsLoadedListener = () => {
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
// ControlNet models loaded - need to remove missing ControlNets from state // ControlNet models loaded - need to remove missing ControlNets from state
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `ControlNet models loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`ControlNet models loaded (${action.payload.ids.length})`
);
selectAllControlNets(getState().controlAdapters).forEach((ca) => { selectAllControlNets(getState().controlAdapters).forEach((ca) => {
const isModelAvailable = some( const isModelAvailable = some(
action.payload.entities, action.payload.entities,
(m) => (m) => m?.model_name === ca?.model?.model_name && m?.base_model === ca?.model?.base_model
m?.model_name === ca?.model?.model_name &&
m?.base_model === ca?.model?.base_model
); );
if (isModelAvailable) { if (isModelAvailable) {
@ -225,17 +179,12 @@ export const addModelsLoadedListener = () => {
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
// ControlNet models loaded - need to remove missing ControlNets from state // ControlNet models loaded - need to remove missing ControlNets from state
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `T2I Adapter models loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`T2I Adapter models loaded (${action.payload.ids.length})`
);
selectAllT2IAdapters(getState().controlAdapters).forEach((ca) => { selectAllT2IAdapters(getState().controlAdapters).forEach((ca) => {
const isModelAvailable = some( const isModelAvailable = some(
action.payload.entities, action.payload.entities,
(m) => (m) => m?.model_name === ca?.model?.model_name && m?.base_model === ca?.model?.base_model
m?.model_name === ca?.model?.model_name &&
m?.base_model === ca?.model?.base_model
); );
if (isModelAvailable) { if (isModelAvailable) {
@ -251,17 +200,12 @@ export const addModelsLoadedListener = () => {
effect: async (action, { getState, dispatch }) => { effect: async (action, { getState, dispatch }) => {
// ControlNet models loaded - need to remove missing ControlNets from state // ControlNet models loaded - need to remove missing ControlNets from state
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `IP Adapter models loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`IP Adapter models loaded (${action.payload.ids.length})`
);
selectAllIPAdapters(getState().controlAdapters).forEach((ca) => { selectAllIPAdapters(getState().controlAdapters).forEach((ca) => {
const isModelAvailable = some( const isModelAvailable = some(
action.payload.entities, action.payload.entities,
(m) => (m) => m?.model_name === ca?.model?.model_name && m?.base_model === ca?.model?.base_model
m?.model_name === ca?.model?.model_name &&
m?.base_model === ca?.model?.base_model
); );
if (isModelAvailable) { if (isModelAvailable) {
@ -276,10 +220,7 @@ export const addModelsLoadedListener = () => {
matcher: modelsApi.endpoints.getTextualInversionModels.matchFulfilled, matcher: modelsApi.endpoints.getTextualInversionModels.matchFulfilled,
effect: async (action) => { effect: async (action) => {
const log = logger('models'); const log = logger('models');
log.info( log.info({ models: action.payload.entities }, `Embeddings loaded (${action.payload.ids.length})`);
{ models: action.payload.entities },
`Embeddings loaded (${action.payload.ids.length})`
);
}, },
}); });
}; };

View File

@ -15,21 +15,12 @@ import { socketConnected } from 'services/events/actions';
import { startAppListening } from '..'; import { startAppListening } from '..';
const matcher = isAnyOf( const matcher = isAnyOf(setPositivePrompt, combinatorialToggled, maxPromptsChanged, maxPromptsReset, socketConnected);
setPositivePrompt,
combinatorialToggled,
maxPromptsChanged,
maxPromptsReset,
socketConnected
);
export const addDynamicPromptsListener = () => { export const addDynamicPromptsListener = () => {
startAppListening({ startAppListening({
matcher, matcher,
effect: async ( effect: async (action, { dispatch, getState, cancelActiveListeners, delay }) => {
action,
{ dispatch, getState, cancelActiveListeners, delay }
) => {
cancelActiveListeners(); cancelActiveListeners();
const state = getState(); const state = getState();
const { positivePrompt } = state.generation; const { positivePrompt } = state.generation;

View File

@ -17,16 +17,9 @@ export const addReceivedOpenAPISchemaListener = () => {
log.debug({ schemaJSON }, 'Received OpenAPI schema'); log.debug({ schemaJSON }, 'Received OpenAPI schema');
const { nodesAllowlist, nodesDenylist } = getState().config; const { nodesAllowlist, nodesDenylist } = getState().config;
const nodeTemplates = parseSchema( const nodeTemplates = parseSchema(schemaJSON, nodesAllowlist, nodesDenylist);
schemaJSON,
nodesAllowlist,
nodesDenylist
);
log.debug( log.debug({ nodeTemplates: parseify(nodeTemplates) }, `Built ${size(nodeTemplates)} node templates`);
{ nodeTemplates: parseify(nodeTemplates) },
`Built ${size(nodeTemplates)} node templates`
);
dispatch(nodeTemplatesBuilt(nodeTemplates)); dispatch(nodeTemplatesBuilt(nodeTemplates));
}, },
@ -36,10 +29,7 @@ export const addReceivedOpenAPISchemaListener = () => {
actionCreator: receivedOpenAPISchema.rejected, actionCreator: receivedOpenAPISchema.rejected,
effect: (action) => { effect: (action) => {
const log = logger('system'); const log = logger('system');
log.error( log.error({ error: parseify(action.error) }, 'Problem retrieving OpenAPI Schema');
{ error: parseify(action.error) },
'Problem retrieving OpenAPI Schema'
);
}, },
}); });
}; };

View File

@ -16,10 +16,7 @@ const $isFirstConnection = atom(true);
export const addSocketConnectedEventListener = () => { export const addSocketConnectedEventListener = () => {
startAppListening({ startAppListening({
actionCreator: socketConnected, actionCreator: socketConnected,
effect: async ( effect: async (action, { dispatch, getState, cancelActiveListeners, delay }) => {
action,
{ dispatch, getState, cancelActiveListeners, delay }
) => {
log.debug('Connected'); log.debug('Connected');
/** /**
@ -86,10 +83,7 @@ export const addSocketConnectedEventListener = () => {
effect: async (action, { dispatch, getState }) => { effect: async (action, { dispatch, getState }) => {
const { nodeTemplates, config } = getState(); const { nodeTemplates, config } = getState();
// We only want to re-fetch the schema if we don't have any node templates // We only want to re-fetch the schema if we don't have any node templates
if ( if (!size(nodeTemplates.templates) && !config.disabledTabs.includes('nodes')) {
!size(nodeTemplates.templates) &&
!config.disabledTabs.includes('nodes')
) {
// This request is a createAsyncThunk - resetting API state as in the above listener // This request is a createAsyncThunk - resetting API state as in the above listener
// will not trigger this request, so we need to manually do it. // will not trigger this request, so we need to manually do it.
dispatch(receivedOpenAPISchema()); dispatch(receivedOpenAPISchema());

View File

@ -1,17 +1,10 @@
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { parseify } from 'common/util/serialize'; import { parseify } from 'common/util/serialize';
import { addImageToStagingArea } from 'features/canvas/store/canvasSlice'; import { addImageToStagingArea } from 'features/canvas/store/canvasSlice';
import { import { boardIdSelected, galleryViewChanged, imageSelected } from 'features/gallery/store/gallerySlice';
boardIdSelected,
galleryViewChanged,
imageSelected,
} from 'features/gallery/store/gallerySlice';
import { IMAGE_CATEGORIES } from 'features/gallery/store/types'; import { IMAGE_CATEGORIES } from 'features/gallery/store/types';
import { isImageOutput } from 'features/nodes/types/common'; import { isImageOutput } from 'features/nodes/types/common';
import { import { LINEAR_UI_OUTPUT, nodeIDDenyList } from 'features/nodes/util/graph/constants';
LINEAR_UI_OUTPUT,
nodeIDDenyList,
} from 'features/nodes/util/graph/constants';
import { boardsApi } from 'services/api/endpoints/boards'; import { boardsApi } from 'services/api/endpoints/boards';
import { imagesApi } from 'services/api/endpoints/images'; import { imagesApi } from 'services/api/endpoints/images';
import { imagesAdapter } from 'services/api/util'; import { imagesAdapter } from 'services/api/util';
@ -29,19 +22,12 @@ export const addInvocationCompleteEventListener = () => {
actionCreator: socketInvocationComplete, actionCreator: socketInvocationComplete,
effect: async (action, { dispatch, getState }) => { effect: async (action, { dispatch, getState }) => {
const { data } = action.payload; const { data } = action.payload;
log.debug( log.debug({ data: parseify(data) }, `Invocation complete (${action.payload.data.node.type})`);
{ data: parseify(data) },
`Invocation complete (${action.payload.data.node.type})`
);
const { result, node, queue_batch_id, source_node_id } = data; const { result, node, queue_batch_id, source_node_id } = data;
// This complete event has an associated image output // This complete event has an associated image output
if ( if (isImageOutput(result) && !nodeTypeDenylist.includes(node.type) && !nodeIDDenyList.includes(source_node_id)) {
isImageOutput(result) &&
!nodeTypeDenylist.includes(node.type) &&
!nodeIDDenyList.includes(source_node_id)
) {
const { image_name } = result.image; const { image_name } = result.image;
const { canvas, gallery } = getState(); const { canvas, gallery } = getState();
@ -56,10 +42,7 @@ export const addInvocationCompleteEventListener = () => {
imageDTORequest.unsubscribe(); imageDTORequest.unsubscribe();
// Add canvas images to the staging area // Add canvas images to the staging area
if ( if (canvas.batchIds.includes(queue_batch_id) && [LINEAR_UI_OUTPUT].includes(data.source_node_id)) {
canvas.batchIds.includes(queue_batch_id) &&
[LINEAR_UI_OUTPUT].includes(data.source_node_id)
) {
dispatch(addImageToStagingArea(imageDTO)); dispatch(addImageToStagingArea(imageDTO));
} }
@ -84,21 +67,13 @@ export const addInvocationCompleteEventListener = () => {
// update the total images for the board // update the total images for the board
dispatch( dispatch(
boardsApi.util.updateQueryData( boardsApi.util.updateQueryData('getBoardImagesTotal', imageDTO.board_id ?? 'none', (draft) => {
'getBoardImagesTotal', // eslint-disable-next-line @typescript-eslint/no-unused-vars
imageDTO.board_id ?? 'none', draft.total += 1;
(draft) => { })
// eslint-disable-next-line @typescript-eslint/no-unused-vars
draft.total += 1;
}
)
); );
dispatch( dispatch(imagesApi.util.invalidateTags([{ type: 'Board', id: imageDTO.board_id ?? 'none' }]));
imagesApi.util.invalidateTags([
{ type: 'Board', id: imageDTO.board_id ?? 'none' },
])
);
const { shouldAutoSwitch } = gallery; const { shouldAutoSwitch } = gallery;
@ -109,10 +84,7 @@ export const addInvocationCompleteEventListener = () => {
dispatch(galleryViewChanged('images')); dispatch(galleryViewChanged('images'));
} }
if ( if (imageDTO.board_id && imageDTO.board_id !== gallery.selectedBoardId) {
imageDTO.board_id &&
imageDTO.board_id !== gallery.selectedBoardId
) {
dispatch( dispatch(
boardIdSelected({ boardIdSelected({
boardId: imageDTO.board_id, boardId: imageDTO.board_id,

View File

@ -9,10 +9,7 @@ export const addInvocationErrorEventListener = () => {
startAppListening({ startAppListening({
actionCreator: socketInvocationError, actionCreator: socketInvocationError,
effect: (action) => { effect: (action) => {
log.error( log.error(action.payload, `Invocation error (${action.payload.data.node.type})`);
action.payload,
`Invocation error (${action.payload.data.node.type})`
);
}, },
}); });
}; };

View File

@ -9,10 +9,7 @@ export const addInvocationRetrievalErrorEventListener = () => {
startAppListening({ startAppListening({
actionCreator: socketInvocationRetrievalError, actionCreator: socketInvocationRetrievalError,
effect: (action) => { effect: (action) => {
log.error( log.error(action.payload, `Invocation retrieval error (${action.payload.data.graph_execution_state_id})`);
action.payload,
`Invocation retrieval error (${action.payload.data.graph_execution_state_id})`
);
}, },
}); });
}; };

View File

@ -9,10 +9,7 @@ export const addInvocationStartedEventListener = () => {
startAppListening({ startAppListening({
actionCreator: socketInvocationStarted, actionCreator: socketInvocationStarted,
effect: (action) => { effect: (action) => {
log.debug( log.debug(action.payload, `Invocation started (${action.payload.data.node.type})`);
action.payload,
`Invocation started (${action.payload.data.node.type})`
);
}, },
}); });
}; };

View File

@ -1,8 +1,5 @@
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { import { socketModelLoadCompleted, socketModelLoadStarted } from 'services/events/actions';
socketModelLoadCompleted,
socketModelLoadStarted,
} from 'services/events/actions';
import { startAppListening } from '../..'; import { startAppListening } from '../..';
@ -12,8 +9,7 @@ export const addModelLoadEventListener = () => {
startAppListening({ startAppListening({
actionCreator: socketModelLoadStarted, actionCreator: socketModelLoadStarted,
effect: (action) => { effect: (action) => {
const { base_model, model_name, model_type, submodel } = const { base_model, model_name, model_type, submodel } = action.payload.data;
action.payload.data;
let message = `Model load started: ${base_model}/${model_type}/${model_name}`; let message = `Model load started: ${base_model}/${model_type}/${model_name}`;
@ -28,8 +24,7 @@ export const addModelLoadEventListener = () => {
startAppListening({ startAppListening({
actionCreator: socketModelLoadCompleted, actionCreator: socketModelLoadCompleted,
effect: (action) => { effect: (action) => {
const { base_model, model_name, model_type, submodel } = const { base_model, model_name, model_type, submodel } = action.payload.data;
action.payload.data;
let message = `Model load complete: ${base_model}/${model_type}/${model_name}`; let message = `Model load complete: ${base_model}/${model_type}/${model_name}`;

View File

@ -13,10 +13,7 @@ export const addSocketQueueItemStatusChangedEventListener = () => {
// we've got new status for the queue item, batch and queue // we've got new status for the queue item, batch and queue
const { queue_item, batch_status, queue_status } = action.payload.data; const { queue_item, batch_status, queue_status } = action.payload.data;
log.debug( log.debug(action.payload, `Queue item ${queue_item.item_id} status updated: ${queue_item.status}`);
action.payload,
`Queue item ${queue_item.item_id} status updated: ${queue_item.status}`
);
// Update this specific queue item in the list of queue items (this is the queue item DTO, without the session) // Update this specific queue item in the list of queue items (this is the queue item DTO, without the session)
dispatch( dispatch(
@ -40,35 +37,23 @@ export const addSocketQueueItemStatusChangedEventListener = () => {
// Update the batch status // Update the batch status
dispatch( dispatch(
queueApi.util.updateQueryData( queueApi.util.updateQueryData('getBatchStatus', { batch_id: batch_status.batch_id }, () => batch_status)
'getBatchStatus',
{ batch_id: batch_status.batch_id },
() => batch_status
)
); );
// Update the queue item status (this is the full queue item, including the session) // Update the queue item status (this is the full queue item, including the session)
dispatch( dispatch(
queueApi.util.updateQueryData( queueApi.util.updateQueryData('getQueueItem', queue_item.item_id, (draft) => {
'getQueueItem', if (!draft) {
queue_item.item_id, return;
(draft) => {
if (!draft) {
return;
}
Object.assign(draft, queue_item);
} }
) Object.assign(draft, queue_item);
})
); );
// Invalidate caches for things we cannot update // Invalidate caches for things we cannot update
// TODO: technically, we could possibly update the current session queue item, but feels safer to just request it again // TODO: technically, we could possibly update the current session queue item, but feels safer to just request it again
dispatch( dispatch(
queueApi.util.invalidateTags([ queueApi.util.invalidateTags(['CurrentSessionQueueItem', 'NextSessionQueueItem', 'InvocationCacheStatus'])
'CurrentSessionQueueItem',
'NextSessionQueueItem',
'InvocationCacheStatus',
])
); );
}, },
}); });

View File

@ -9,10 +9,7 @@ export const addSessionRetrievalErrorEventListener = () => {
startAppListening({ startAppListening({
actionCreator: socketSessionRetrievalError, actionCreator: socketSessionRetrievalError,
effect: (action) => { effect: (action) => {
log.error( log.error(action.payload, `Session retrieval error (${action.payload.data.graph_execution_state_id})`);
action.payload,
`Session retrieval error (${action.payload.data.graph_execution_state_id})`
);
}, },
}); });
}; };

View File

@ -3,10 +3,7 @@ import { updateAllNodesRequested } from 'features/nodes/store/actions';
import { nodeReplaced } from 'features/nodes/store/nodesSlice'; import { nodeReplaced } from 'features/nodes/store/nodesSlice';
import { NodeUpdateError } from 'features/nodes/types/error'; import { NodeUpdateError } from 'features/nodes/types/error';
import { isInvocationNode } from 'features/nodes/types/invocation'; import { isInvocationNode } from 'features/nodes/types/invocation';
import { import { getNeedsUpdate, updateNode } from 'features/nodes/util/node/nodeUpdate';
getNeedsUpdate,
updateNode,
} from 'features/nodes/util/node/nodeUpdate';
import { addToast } from 'features/system/store/systemSlice'; import { addToast } from 'features/system/store/systemSlice';
import { makeToast } from 'features/system/util/makeToast'; import { makeToast } from 'features/system/util/makeToast';
import { t } from 'i18next'; import { t } from 'i18next';

View File

@ -10,9 +10,7 @@ import type { BatchConfig, ImageDTO } from 'services/api/types';
import { startAppListening } from '..'; import { startAppListening } from '..';
export const upscaleRequested = createAction<{ imageDTO: ImageDTO }>( export const upscaleRequested = createAction<{ imageDTO: ImageDTO }>(`upscale/upscaleRequested`);
`upscale/upscaleRequested`
);
export const addUpscaleRequestedListener = () => { export const addUpscaleRequestedListener = () => {
startAppListening({ startAppListening({
@ -24,8 +22,7 @@ export const addUpscaleRequestedListener = () => {
const { image_name } = imageDTO; const { image_name } = imageDTO;
const state = getState(); const state = getState();
const { isAllowedToUpscale, detailTKey } = const { isAllowedToUpscale, detailTKey } = createIsAllowedToUpscaleSelector(imageDTO)(state);
createIsAllowedToUpscaleSelector(imageDTO)(state);
// if we can't upscale, show a toast and return // if we can't upscale, show a toast and return
if (!isAllowedToUpscale) { if (!isAllowedToUpscale) {
@ -66,21 +63,11 @@ export const addUpscaleRequestedListener = () => {
const enqueueResult = await req.unwrap(); const enqueueResult = await req.unwrap();
req.reset(); req.reset();
log.debug( log.debug({ enqueueResult: parseify(enqueueResult) }, t('queue.graphQueued'));
{ enqueueResult: parseify(enqueueResult) },
t('queue.graphQueued')
);
} catch (error) { } catch (error) {
log.error( log.error({ enqueueBatchArg: parseify(enqueueBatchArg) }, t('queue.graphFailedToQueue'));
{ enqueueBatchArg: parseify(enqueueBatchArg) },
t('queue.graphFailedToQueue')
);
if ( if (error instanceof Object && 'status' in error && error.status === 403) {
error instanceof Object &&
'status' in error &&
error.status === 403
) {
return; return;
} else { } else {
dispatch( dispatch(

View File

@ -1,14 +1,8 @@
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { parseify } from 'common/util/serialize'; import { parseify } from 'common/util/serialize';
import { import { workflowLoaded, workflowLoadRequested } from 'features/nodes/store/actions';
workflowLoaded,
workflowLoadRequested,
} from 'features/nodes/store/actions';
import { $flow } from 'features/nodes/store/reactFlowInstance'; import { $flow } from 'features/nodes/store/reactFlowInstance';
import { import { WorkflowMigrationError, WorkflowVersionError } from 'features/nodes/types/error';
WorkflowMigrationError,
WorkflowVersionError,
} from 'features/nodes/types/error';
import { validateWorkflow } from 'features/nodes/util/workflow/validateWorkflow'; import { validateWorkflow } from 'features/nodes/util/workflow/validateWorkflow';
import { addToast } from 'features/system/store/systemSlice'; import { addToast } from 'features/system/store/systemSlice';
import { makeToast } from 'features/system/util/makeToast'; import { makeToast } from 'features/system/util/makeToast';
@ -28,10 +22,7 @@ export const addWorkflowLoadRequestedListener = () => {
const nodeTemplates = getState().nodeTemplates.templates; const nodeTemplates = getState().nodeTemplates.templates;
try { try {
const { workflow: validatedWorkflow, warnings } = validateWorkflow( const { workflow: validatedWorkflow, warnings } = validateWorkflow(workflow, nodeTemplates);
workflow,
nodeTemplates
);
if (asCopy) { if (asCopy) {
// If we're loading a copy, we need to remove the ID so that the backend will create a new workflow // If we're loading a copy, we need to remove the ID so that the backend will create a new workflow
@ -108,10 +99,7 @@ export const addWorkflowLoadRequestedListener = () => {
); );
} else { } else {
// Some other error occurred // Some other error occurred
log.error( log.error({ error: parseify(e) }, t('nodes.unknownErrorValidatingWorkflow'));
{ error: parseify(e) },
t('nodes.unknownErrorValidatingWorkflow')
);
dispatch( dispatch(
addToast( addToast(
makeToast({ makeToast({

View File

@ -8,6 +8,4 @@ declare global {
} }
} }
export const $store = atom< export const $store = atom<Readonly<ReturnType<typeof createStore>> | undefined>();
Readonly<ReturnType<typeof createStore>> | undefined
>();

View File

@ -1,7 +1,4 @@
import type { WorkflowCategory } from 'features/nodes/types/workflow'; import type { WorkflowCategory } from 'features/nodes/types/workflow';
import { atom } from 'nanostores'; import { atom } from 'nanostores';
export const $workflowCategories = atom<WorkflowCategory[]>([ export const $workflowCategories = atom<WorkflowCategory[]>(['user', 'default']);
'user',
'default',
]);

View File

@ -1,17 +1,10 @@
import type { ThunkDispatch, UnknownAction } from '@reduxjs/toolkit'; import type { ThunkDispatch, UnknownAction } from '@reduxjs/toolkit';
import { import { autoBatchEnhancer, combineReducers, configureStore } from '@reduxjs/toolkit';
autoBatchEnhancer,
combineReducers,
configureStore,
} from '@reduxjs/toolkit';
import { logger } from 'app/logging/logger'; import { logger } from 'app/logging/logger';
import { idbKeyValDriver } from 'app/store/enhancers/reduxRemember/driver'; import { idbKeyValDriver } from 'app/store/enhancers/reduxRemember/driver';
import { errorHandler } from 'app/store/enhancers/reduxRemember/errors'; import { errorHandler } from 'app/store/enhancers/reduxRemember/errors';
import { canvasPersistDenylist } from 'features/canvas/store/canvasPersistDenylist'; import { canvasPersistDenylist } from 'features/canvas/store/canvasPersistDenylist';
import canvasReducer, { import canvasReducer, { initialCanvasState, migrateCanvasState } from 'features/canvas/store/canvasSlice';
initialCanvasState,
migrateCanvasState,
} from 'features/canvas/store/canvasSlice';
import changeBoardModalReducer from 'features/changeBoardModal/store/slice'; import changeBoardModalReducer from 'features/changeBoardModal/store/slice';
import { controlAdaptersPersistDenylist } from 'features/controlAdapters/store/controlAdaptersPersistDenylist'; import { controlAdaptersPersistDenylist } from 'features/controlAdapters/store/controlAdaptersPersistDenylist';
import controlAdaptersReducer, { import controlAdaptersReducer, {
@ -25,32 +18,17 @@ import dynamicPromptsReducer, {
migrateDynamicPromptsState, migrateDynamicPromptsState,
} from 'features/dynamicPrompts/store/dynamicPromptsSlice'; } from 'features/dynamicPrompts/store/dynamicPromptsSlice';
import { galleryPersistDenylist } from 'features/gallery/store/galleryPersistDenylist'; import { galleryPersistDenylist } from 'features/gallery/store/galleryPersistDenylist';
import galleryReducer, { import galleryReducer, { initialGalleryState, migrateGalleryState } from 'features/gallery/store/gallerySlice';
initialGalleryState, import hrfReducer, { initialHRFState, migrateHRFState } from 'features/hrf/store/hrfSlice';
migrateGalleryState, import loraReducer, { initialLoraState, migrateLoRAState } from 'features/lora/store/loraSlice';
} from 'features/gallery/store/gallerySlice';
import hrfReducer, {
initialHRFState,
migrateHRFState,
} from 'features/hrf/store/hrfSlice';
import loraReducer, {
initialLoraState,
migrateLoRAState,
} from 'features/lora/store/loraSlice';
import modelmanagerReducer, { import modelmanagerReducer, {
initialModelManagerState, initialModelManagerState,
migrateModelManagerState, migrateModelManagerState,
} from 'features/modelManager/store/modelManagerSlice'; } from 'features/modelManager/store/modelManagerSlice';
import { nodesPersistDenylist } from 'features/nodes/store/nodesPersistDenylist'; import { nodesPersistDenylist } from 'features/nodes/store/nodesPersistDenylist';
import nodesReducer, { import nodesReducer, { initialNodesState, migrateNodesState } from 'features/nodes/store/nodesSlice';
initialNodesState,
migrateNodesState,
} from 'features/nodes/store/nodesSlice';
import nodeTemplatesReducer from 'features/nodes/store/nodeTemplatesSlice'; import nodeTemplatesReducer from 'features/nodes/store/nodeTemplatesSlice';
import workflowReducer, { import workflowReducer, { initialWorkflowState, migrateWorkflowState } from 'features/nodes/store/workflowSlice';
initialWorkflowState,
migrateWorkflowState,
} from 'features/nodes/store/workflowSlice';
import { generationPersistDenylist } from 'features/parameters/store/generationPersistDenylist'; import { generationPersistDenylist } from 'features/parameters/store/generationPersistDenylist';
import generationReducer, { import generationReducer, {
initialGenerationState, initialGenerationState,
@ -62,21 +40,12 @@ import postprocessingReducer, {
migratePostprocessingState, migratePostprocessingState,
} from 'features/parameters/store/postprocessingSlice'; } from 'features/parameters/store/postprocessingSlice';
import queueReducer from 'features/queue/store/queueSlice'; import queueReducer from 'features/queue/store/queueSlice';
import sdxlReducer, { import sdxlReducer, { initialSDXLState, migrateSDXLState } from 'features/sdxl/store/sdxlSlice';
initialSDXLState,
migrateSDXLState,
} from 'features/sdxl/store/sdxlSlice';
import configReducer from 'features/system/store/configSlice'; import configReducer from 'features/system/store/configSlice';
import { systemPersistDenylist } from 'features/system/store/systemPersistDenylist'; import { systemPersistDenylist } from 'features/system/store/systemPersistDenylist';
import systemReducer, { import systemReducer, { initialSystemState, migrateSystemState } from 'features/system/store/systemSlice';
initialSystemState,
migrateSystemState,
} from 'features/system/store/systemSlice';
import { uiPersistDenylist } from 'features/ui/store/uiPersistDenylist'; import { uiPersistDenylist } from 'features/ui/store/uiPersistDenylist';
import uiReducer, { import uiReducer, { initialUIState, migrateUIState } from 'features/ui/store/uiSlice';
initialUIState,
migrateUIState,
} from 'features/ui/store/uiSlice';
import { diff } from 'jsondiffpatch'; import { diff } from 'jsondiffpatch';
import { defaultsDeep, keys, omit, pick } from 'lodash-es'; import { defaultsDeep, keys, omit, pick } from 'lodash-es';
import dynamicMiddlewares from 'redux-dynamic-middlewares'; import dynamicMiddlewares from 'redux-dynamic-middlewares';
@ -206,10 +175,7 @@ const unserialize: UnserializeFunction = (data, key) => {
); );
return transformed; return transformed;
} catch (err) { } catch (err) {
log.warn( log.warn({ error: serializeError(err) }, `Error rehydrating slice "${key}", falling back to default initial state`);
{ error: serializeError(err) },
`Error rehydrating slice "${key}", falling back to default initial state`
);
return config.initialState; return config.initialState;
} }
}; };
@ -229,10 +195,7 @@ const serializationDenylist: {
}; };
export const serialize: SerializeFunction = (data, key) => { export const serialize: SerializeFunction = (data, key) => {
const result = omit( const result = omit(data, serializationDenylist[key as keyof typeof serializationDenylist] ?? []);
data,
serializationDenylist[key as keyof typeof serializationDenylist] ?? []
);
return JSON.stringify(result); return JSON.stringify(result);
}; };
@ -256,9 +219,7 @@ export const createStore = (uniqueStoreKey?: string, persist = true) =>
persistDebounce: 300, persistDebounce: 300,
serialize, serialize,
unserialize, unserialize,
prefix: uniqueStoreKey prefix: uniqueStoreKey ? `${STORAGE_PREFIX}${uniqueStoreKey}-` : STORAGE_PREFIX,
? `${STORAGE_PREFIX}${uniqueStoreKey}-`
: STORAGE_PREFIX,
errorHandler, errorHandler,
}) })
); );
@ -278,9 +239,7 @@ export const createStore = (uniqueStoreKey?: string, persist = true) =>
}, },
}); });
export type AppGetState = ReturnType< export type AppGetState = ReturnType<ReturnType<typeof createStore>['getState']>;
ReturnType<typeof createStore>['getState']
>;
export type RootState = ReturnType<ReturnType<typeof createStore>['getState']>; export type RootState = ReturnType<ReturnType<typeof createStore>['getState']>;
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
export type AppThunkDispatch = ThunkDispatch<RootState, any, UnknownAction>; export type AppThunkDispatch = ThunkDispatch<RootState, any, UnknownAction>;

View File

@ -1,17 +1,9 @@
import type { ChakraProps } from '@invoke-ai/ui-library'; import type { ChakraProps } from '@invoke-ai/ui-library';
import { import { CompositeNumberInput, Flex, FormControl, FormLabel } from '@invoke-ai/ui-library';
CompositeNumberInput,
Flex,
FormControl,
FormLabel,
} from '@invoke-ai/ui-library';
import type { CSSProperties } from 'react'; import type { CSSProperties } from 'react';
import { memo, useCallback } from 'react'; import { memo, useCallback } from 'react';
import { RgbaColorPicker } from 'react-colorful'; import { RgbaColorPicker } from 'react-colorful';
import type { import type { ColorPickerBaseProps, RgbaColor } from 'react-colorful/dist/types';
ColorPickerBaseProps,
RgbaColor,
} from 'react-colorful/dist/types';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
type IAIColorPickerProps = ColorPickerBaseProps<RgbaColor> & { type IAIColorPickerProps = ColorPickerBaseProps<RgbaColor> & {
@ -39,30 +31,13 @@ const numberInputWidth: ChakraProps['w'] = '4.2rem';
const IAIColorPicker = (props: IAIColorPickerProps) => { const IAIColorPicker = (props: IAIColorPickerProps) => {
const { color, onChange, withNumberInput, ...rest } = props; const { color, onChange, withNumberInput, ...rest } = props;
const { t } = useTranslation(); const { t } = useTranslation();
const handleChangeR = useCallback( const handleChangeR = useCallback((r: number) => onChange({ ...color, r }), [color, onChange]);
(r: number) => onChange({ ...color, r }), const handleChangeG = useCallback((g: number) => onChange({ ...color, g }), [color, onChange]);
[color, onChange] const handleChangeB = useCallback((b: number) => onChange({ ...color, b }), [color, onChange]);
); const handleChangeA = useCallback((a: number) => onChange({ ...color, a }), [color, onChange]);
const handleChangeG = useCallback(
(g: number) => onChange({ ...color, g }),
[color, onChange]
);
const handleChangeB = useCallback(
(b: number) => onChange({ ...color, b }),
[color, onChange]
);
const handleChangeA = useCallback(
(a: number) => onChange({ ...color, a }),
[color, onChange]
);
return ( return (
<Flex sx={sx}> <Flex sx={sx}>
<RgbaColorPicker <RgbaColorPicker color={color} onChange={onChange} style={colorPickerStyles} {...rest} />
color={color}
onChange={onChange}
style={colorPickerStyles}
{...rest}
/>
{withNumberInput && ( {withNumberInput && (
<Flex> <Flex>
<FormControl> <FormControl>

View File

@ -1,26 +1,11 @@
import type { import type { ChakraProps, FlexProps, SystemStyleObject } from '@invoke-ai/ui-library';
ChakraProps,
FlexProps,
SystemStyleObject,
} from '@invoke-ai/ui-library';
import { Flex, Icon, Image } from '@invoke-ai/ui-library'; import { Flex, Icon, Image } from '@invoke-ai/ui-library';
import { import { IAILoadingImageFallback, IAINoContentFallback } from 'common/components/IAIImageFallback';
IAILoadingImageFallback,
IAINoContentFallback,
} from 'common/components/IAIImageFallback';
import ImageMetadataOverlay from 'common/components/ImageMetadataOverlay'; import ImageMetadataOverlay from 'common/components/ImageMetadataOverlay';
import { useImageUploadButton } from 'common/hooks/useImageUploadButton'; import { useImageUploadButton } from 'common/hooks/useImageUploadButton';
import type { import type { TypesafeDraggableData, TypesafeDroppableData } from 'features/dnd/types';
TypesafeDraggableData,
TypesafeDroppableData,
} from 'features/dnd/types';
import ImageContextMenu from 'features/gallery/components/ImageContextMenu/ImageContextMenu'; import ImageContextMenu from 'features/gallery/components/ImageContextMenu/ImageContextMenu';
import type { import type { MouseEvent, ReactElement, ReactNode, SyntheticEvent } from 'react';
MouseEvent,
ReactElement,
ReactNode,
SyntheticEvent,
} from 'react';
import { memo, useCallback, useMemo, useState } from 'react'; import { memo, useCallback, useMemo, useState } from 'react';
import { PiImageBold, PiUploadSimpleBold } from 'react-icons/pi'; import { PiImageBold, PiUploadSimpleBold } from 'react-icons/pi';
import type { ImageDTO, PostUploadAction } from 'services/api/types'; import type { ImageDTO, PostUploadAction } from 'services/api/types';
@ -165,14 +150,8 @@ const IAIDndImage = (props: IAIDndImageProps) => {
<Image <Image
src={thumbnail ? imageDTO.thumbnail_url : imageDTO.image_url} src={thumbnail ? imageDTO.thumbnail_url : imageDTO.image_url}
fallbackStrategy="beforeLoadOrError" fallbackStrategy="beforeLoadOrError"
fallbackSrc={ fallbackSrc={useThumbailFallback ? imageDTO.thumbnail_url : undefined}
useThumbailFallback ? imageDTO.thumbnail_url : undefined fallback={useThumbailFallback ? undefined : <IAILoadingImageFallback image={imageDTO} />}
}
fallback={
useThumbailFallback ? undefined : (
<IAILoadingImageFallback image={imageDTO} />
)
}
onError={onError} onError={onError}
draggable={false} draggable={false}
w={imageDTO.width} w={imageDTO.width}
@ -183,13 +162,8 @@ const IAIDndImage = (props: IAIDndImageProps) => {
sx={imageSx} sx={imageSx}
data-testid={dataTestId} data-testid={dataTestId}
/> />
{withMetadataOverlay && ( {withMetadataOverlay && <ImageMetadataOverlay imageDTO={imageDTO} />}
<ImageMetadataOverlay imageDTO={imageDTO} /> <SelectionOverlay isSelected={isSelected} isHovered={withHoverOverlay ? isHovered : false} />
)}
<SelectionOverlay
isSelected={isSelected}
isHovered={withHoverOverlay ? isHovered : false}
/>
</Flex> </Flex>
)} )}
{!imageDTO && !isUploadDisabled && ( {!imageDTO && !isUploadDisabled && (
@ -202,20 +176,10 @@ const IAIDndImage = (props: IAIDndImageProps) => {
)} )}
{!imageDTO && isUploadDisabled && noContentFallback} {!imageDTO && isUploadDisabled && noContentFallback}
{imageDTO && !isDragDisabled && ( {imageDTO && !isDragDisabled && (
<IAIDraggable <IAIDraggable data={draggableData} disabled={isDragDisabled || !imageDTO} onClick={onClick} />
data={draggableData}
disabled={isDragDisabled || !imageDTO}
onClick={onClick}
/>
)} )}
{children} {children}
{!isDropDisabled && ( {!isDropDisabled && <IAIDroppable data={droppableData} disabled={isDropDisabled} dropLabel={dropLabel} />}
<IAIDroppable
data={droppableData}
disabled={isDropDisabled}
dropLabel={dropLabel}
/>
)}
</Flex> </Flex>
)} )}
</ImageContextMenu> </ImageContextMenu>

View File

@ -27,12 +27,7 @@ const IAIDropOverlay = (props: Props) => {
const { isOver, label = t('gallery.drop') } = props; const { isOver, label = t('gallery.drop') } = props;
const motionId = useRef(uuidv4()); const motionId = useRef(uuidv4());
return ( return (
<motion.div <motion.div key={motionId.current} initial={initial} animate={animate} exit={exit}>
key={motionId.current}
initial={initial}
animate={animate}
exit={exit}
>
<Flex position="absolute" top={0} insetInlineStart={0} w="full" h="full"> <Flex position="absolute" top={0} insetInlineStart={0} w="full" h="full">
<Flex <Flex
position="absolute" position="absolute"

View File

@ -36,9 +36,7 @@ const IAIDroppable = (props: IAIDroppableProps) => {
pointerEvents={active ? 'auto' : 'none'} pointerEvents={active ? 'auto' : 'none'}
> >
<AnimatePresence> <AnimatePresence>
{isValidDrop(data, active) && ( {isValidDrop(data, active) && <IAIDropOverlay isOver={isOver} label={dropLabel} />}
<IAIDropOverlay isOver={isOver} label={dropLabel} />
)}
</AnimatePresence> </AnimatePresence>
</Box> </Box>
); );

View File

@ -16,13 +16,7 @@ const skeletonStyles: SystemStyleObject = {
const IAIFillSkeleton = () => { const IAIFillSkeleton = () => {
return ( return (
<Skeleton sx={skeletonStyles}> <Skeleton sx={skeletonStyles}>
<Box <Box position="absolute" top={0} insetInlineStart={0} height="full" width="full" />
position="absolute"
top={0}
insetInlineStart={0}
height="full"
width="full"
/>
</Skeleton> </Skeleton>
); );
}; };

View File

@ -19,15 +19,7 @@ export const IAILoadingImageFallback = memo((props: Props) => {
} }
return ( return (
<Flex <Flex opacity={0.7} w="full" h="full" alignItems="center" justifyContent="center" borderRadius="base" bg="base.900">
opacity={0.7}
w="full"
h="full"
alignItems="center"
justifyContent="center"
borderRadius="base"
bg="base.900"
>
<Spinner size="xl" /> <Spinner size="xl" />
</Flex> </Flex>
); );
@ -77,32 +69,30 @@ type IAINoImageFallbackWithSpinnerProps = FlexProps & {
label?: string; label?: string;
}; };
export const IAINoContentFallbackWithSpinner = memo( export const IAINoContentFallbackWithSpinner = memo((props: IAINoImageFallbackWithSpinnerProps) => {
(props: IAINoImageFallbackWithSpinnerProps) => { const { sx, ...rest } = props;
const { sx, ...rest } = props; const styles = useMemo(
const styles = useMemo( () => ({
() => ({ w: 'full',
w: 'full', h: 'full',
h: 'full', alignItems: 'center',
alignItems: 'center', justifyContent: 'center',
justifyContent: 'center', borderRadius: 'base',
borderRadius: 'base', flexDir: 'column',
flexDir: 'column', gap: 2,
gap: 2, userSelect: 'none',
userSelect: 'none', opacity: 0.7,
opacity: 0.7, color: 'base.500',
color: 'base.500', ...sx,
...sx, }),
}), [sx]
[sx] );
);
return ( return (
<Flex sx={styles} {...rest}> <Flex sx={styles} {...rest}>
<Spinner size="xl" /> <Spinner size="xl" />
{props.label && <Text textAlign="center">{props.label}</Text>} {props.label && <Text textAlign="center">{props.label}</Text>}
</Flex> </Flex>
); );
} });
);
IAINoContentFallbackWithSpinner.displayName = 'IAINoContentFallbackWithSpinner'; IAINoContentFallbackWithSpinner.displayName = 'IAINoContentFallbackWithSpinner';

View File

@ -91,9 +91,7 @@ const ImageUploadOverlay = (props: ImageUploadOverlayProps) => {
) : ( ) : (
<> <>
<Heading size="lg">{t('toast.invalidUpload')}</Heading> <Heading size="lg">{t('toast.invalidUpload')}</Heading>
<Heading size="md"> <Heading size="md">{t('toast.uploadFailedInvalidUploadDesc')}</Heading>
{t('toast.uploadFailedInvalidUploadDesc')}
</Heading>
</> </>
)} )}
</Flex> </Flex>

View File

@ -28,46 +28,39 @@ type Props = {
children: ReactElement; children: ReactElement;
}; };
export const InformationalPopover = memo( export const InformationalPopover = memo(({ feature, children, inPortal = true, ...rest }: Props) => {
({ feature, children, inPortal = true, ...rest }: Props) => { const shouldEnableInformationalPopovers = useAppSelector((s) => s.system.shouldEnableInformationalPopovers);
const shouldEnableInformationalPopovers = useAppSelector(
(s) => s.system.shouldEnableInformationalPopovers
);
const data = useMemo(() => POPOVER_DATA[feature], [feature]); const data = useMemo(() => POPOVER_DATA[feature], [feature]);
const popoverProps = useMemo( const popoverProps = useMemo(() => merge(omit(data, ['image', 'href', 'buttonLabel']), rest), [data, rest]);
() => merge(omit(data, ['image', 'href', 'buttonLabel']), rest),
[data, rest]
);
if (!shouldEnableInformationalPopovers) { if (!shouldEnableInformationalPopovers) {
return children; return children;
}
return (
<Popover
isLazy
closeOnBlur={false}
trigger="hover"
variant="informational"
openDelay={OPEN_DELAY}
modifiers={POPPER_MODIFIERS}
placement="top"
{...popoverProps}
>
<PopoverTrigger>{children}</PopoverTrigger>
{inPortal ? (
<Portal>
<Content data={data} feature={feature} />
</Portal>
) : (
<Content data={data} feature={feature} />
)}
</Popover>
);
} }
);
return (
<Popover
isLazy
closeOnBlur={false}
trigger="hover"
variant="informational"
openDelay={OPEN_DELAY}
modifiers={POPPER_MODIFIERS}
placement="top"
{...popoverProps}
>
<PopoverTrigger>{children}</PopoverTrigger>
{inPortal ? (
<Portal>
<Content data={data} feature={feature} />
</Portal>
) : (
<Content data={data} feature={feature} />
)}
</Popover>
);
});
InformationalPopover.displayName = 'InformationalPopover'; InformationalPopover.displayName = 'InformationalPopover';
@ -79,10 +72,7 @@ type ContentProps = {
const Content = ({ data, feature }: ContentProps) => { const Content = ({ data, feature }: ContentProps) => {
const { t } = useTranslation(); const { t } = useTranslation();
const heading = useMemo<string | undefined>( const heading = useMemo<string | undefined>(() => t(`popovers.${feature}.heading`), [feature, t]);
() => t(`popovers.${feature}.heading`),
[feature, t]
);
const paragraphs = useMemo<string[]>( const paragraphs = useMemo<string[]>(
() => () =>

View File

@ -95,6 +95,4 @@ export const POPOVER_DATA: { [key in Feature]?: PopoverData } = {
export const OPEN_DELAY = 1000; // in milliseconds export const OPEN_DELAY = 1000; // in milliseconds
export const POPPER_MODIFIERS: PopoverProps['modifiers'] = [ export const POPPER_MODIFIERS: PopoverProps['modifiers'] = [{ name: 'preventOverflow', options: { padding: 10 } }];
{ name: 'preventOverflow', options: { padding: 10 } },
];

View File

@ -6,14 +6,7 @@ import { memo } from 'react';
const Loading = () => { const Loading = () => {
return ( return (
<Flex <Flex position="relative" width="100vw" height="100vh" alignItems="center" justifyContent="center" bg="#151519">
position="relative"
width="100vw"
height="100vh"
alignItems="center"
justifyContent="center"
bg="#151519"
>
<Image src={InvokeLogoWhite} w="8rem" h="8rem" /> <Image src={InvokeLogoWhite} w="8rem" h="8rem" />
<Spinner <Spinner
label="Loading" label="Loading"

View File

@ -13,12 +13,7 @@ type Props = PropsWithChildren & {
const styles: CSSProperties = { height: '100%', width: '100%' }; const styles: CSSProperties = { height: '100%', width: '100%' };
const ScrollableContent = ({ const ScrollableContent = ({ children, maxHeight, overflowX = 'hidden', overflowY = 'scroll' }: Props) => {
children,
maxHeight,
overflowX = 'hidden',
overflowY = 'scroll',
}: Props) => {
const overlayscrollbarsOptions = useMemo( const overlayscrollbarsOptions = useMemo(
() => getOverlayScrollbarsParams(overflowX, overflowY).options, () => getOverlayScrollbarsParams(overflowX, overflowY).options,
[overflowX, overflowY] [overflowX, overflowY]
@ -26,11 +21,7 @@ const ScrollableContent = ({
return ( return (
<Flex w="full" h="full" maxHeight={maxHeight} position="relative"> <Flex w="full" h="full" maxHeight={maxHeight} position="relative">
<Box position="absolute" top={0} left={0} right={0} bottom={0}> <Box position="absolute" top={0} left={0} right={0} bottom={0}>
<OverlayScrollbarsComponent <OverlayScrollbarsComponent defer style={styles} options={overlayscrollbarsOptions}>
defer
style={styles}
options={overlayscrollbarsOptions}
>
{children} {children}
</OverlayScrollbarsComponent> </OverlayScrollbarsComponent>
</Box> </Box>

View File

@ -14,22 +14,19 @@ const accept: Accept = {
'image/jpeg': ['.jpg', '.jpeg', '.png'], 'image/jpeg': ['.jpg', '.jpeg', '.png'],
}; };
const selectPostUploadAction = createMemoizedSelector( const selectPostUploadAction = createMemoizedSelector(activeTabNameSelector, (activeTabName) => {
activeTabNameSelector, let postUploadAction: PostUploadAction = { type: 'TOAST' };
(activeTabName) => {
let postUploadAction: PostUploadAction = { type: 'TOAST' };
if (activeTabName === 'unifiedCanvas') { if (activeTabName === 'unifiedCanvas') {
postUploadAction = { type: 'SET_CANVAS_INITIAL_IMAGE' }; postUploadAction = { type: 'SET_CANVAS_INITIAL_IMAGE' };
}
if (activeTabName === 'img2img') {
postUploadAction = { type: 'SET_INITIAL_IMAGE' };
}
return postUploadAction;
} }
);
if (activeTabName === 'img2img') {
postUploadAction = { type: 'SET_INITIAL_IMAGE' };
}
return postUploadAction;
});
export const useFullscreenDropzone = () => { export const useFullscreenDropzone = () => {
const { t } = useTranslation(); const { t } = useTranslation();
@ -112,9 +109,7 @@ export const useFullscreenDropzone = () => {
// Set the files on the dropzone.inputRef // Set the files on the dropzone.inputRef
dropzone.inputRef.current.files = e.clipboardData.files; dropzone.inputRef.current.files = e.clipboardData.files;
// Dispatch the change event, dropzone catches this and we get to use its own validation // Dispatch the change event, dropzone catches this and we get to use its own validation
dropzone.inputRef.current?.dispatchEvent( dropzone.inputRef.current?.dispatchEvent(new Event('change', { bubbles: true }));
new Event('change', { bubbles: true })
);
} }
}; };

View File

@ -9,13 +9,8 @@ import { useHotkeys } from 'react-hotkeys-hook';
export const useGlobalHotkeys = () => { export const useGlobalHotkeys = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const isModelManagerEnabled = const isModelManagerEnabled = useFeatureStatus('modelManager').isFeatureEnabled;
useFeatureStatus('modelManager').isFeatureEnabled; const { queueBack, isDisabled: isDisabledQueueBack, isLoading: isLoadingQueueBack } = useQueueBack();
const {
queueBack,
isDisabled: isDisabledQueueBack,
isLoading: isLoadingQueueBack,
} = useQueueBack();
useHotkeys( useHotkeys(
['ctrl+enter', 'meta+enter'], ['ctrl+enter', 'meta+enter'],
@ -28,11 +23,7 @@ export const useGlobalHotkeys = () => {
[queueBack, isDisabledQueueBack, isLoadingQueueBack] [queueBack, isDisabledQueueBack, isLoadingQueueBack]
); );
const { const { queueFront, isDisabled: isDisabledQueueFront, isLoading: isLoadingQueueFront } = useQueueFront();
queueFront,
isDisabled: isDisabledQueueFront,
isLoading: isLoadingQueueFront,
} = useQueueFront();
useHotkeys( useHotkeys(
['ctrl+shift+enter', 'meta+shift+enter'], ['ctrl+shift+enter', 'meta+shift+enter'],
@ -61,11 +52,7 @@ export const useGlobalHotkeys = () => {
[cancelQueueItem, isDisabledCancelQueueItem, isLoadingCancelQueueItem] [cancelQueueItem, isDisabledCancelQueueItem, isLoadingCancelQueueItem]
); );
const { const { clearQueue, isDisabled: isDisabledClearQueue, isLoading: isLoadingClearQueue } = useClearQueue();
clearQueue,
isDisabled: isDisabledClearQueue,
isLoading: isLoadingClearQueue,
} = useClearQueue();
useHotkeys( useHotkeys(
['ctrl+shift+x', 'meta+shift+x'], ['ctrl+shift+x', 'meta+shift+x'],

View File

@ -28,11 +28,8 @@ export const useGroupedModelCombobox = <T extends AnyModelConfigEntity>(
arg: UseGroupedModelComboboxArg<T> arg: UseGroupedModelComboboxArg<T>
): UseGroupedModelComboboxReturn => { ): UseGroupedModelComboboxReturn => {
const { t } = useTranslation(); const { t } = useTranslation();
const base_model = useAppSelector( const base_model = useAppSelector((s) => s.generation.model?.base_model ?? 'sdxl');
(s) => s.generation.model?.base_model ?? 'sdxl' const { modelEntities, selectedModel, getIsDisabled, onChange, isLoading } = arg;
);
const { modelEntities, selectedModel, getIsDisabled, onChange, isLoading } =
arg;
const options = useMemo<GroupBase<ComboboxOption>[]>(() => { const options = useMemo<GroupBase<ComboboxOption>[]>(() => {
if (!modelEntities) { if (!modelEntities) {
return []; return [];
@ -60,11 +57,8 @@ export const useGroupedModelCombobox = <T extends AnyModelConfigEntity>(
const value = useMemo( const value = useMemo(
() => () =>
options options.flatMap((o) => o.options).find((m) => (selectedModel ? m.value === getModelId(selectedModel) : false)) ??
.flatMap((o) => o.options) null,
.find((m) =>
selectedModel ? m.value === getModelId(selectedModel) : false
) ?? null,
[options, selectedModel] [options, selectedModel]
); );

View File

@ -28,10 +28,7 @@ type UseImageUploadButtonArgs = {
* <Button {...getUploadButtonProps()} /> // will open the file dialog on click * <Button {...getUploadButtonProps()} /> // will open the file dialog on click
* <input {...getUploadInputProps()} /> // hidden, handles native upload functionality * <input {...getUploadInputProps()} /> // hidden, handles native upload functionality
*/ */
export const useImageUploadButton = ({ export const useImageUploadButton = ({ postUploadAction, isDisabled }: UseImageUploadButtonArgs) => {
postUploadAction,
isDisabled,
}: UseImageUploadButtonArgs) => {
const autoAddBoardId = useAppSelector((s) => s.gallery.autoAddBoardId); const autoAddBoardId = useAppSelector((s) => s.gallery.autoAddBoardId);
const [uploadImage] = useUploadImageMutation(); const [uploadImage] = useUploadImageMutation();
const onDropAccepted = useCallback( const onDropAccepted = useCallback(

View File

@ -27,15 +27,7 @@ const selector = createMemoizedSelector(
selectDynamicPromptsSlice, selectDynamicPromptsSlice,
activeTabNameSelector, activeTabNameSelector,
], ],
( (controlAdapters, generation, system, nodes, nodeTemplates, dynamicPrompts, activeTabName) => {
controlAdapters,
generation,
system,
nodes,
nodeTemplates,
dynamicPrompts,
activeTabName
) => {
const { initialImage, model, positivePrompt } = generation; const { initialImage, model, positivePrompt } = generation;
const { isConnected } = system; const { isConnected } = system;
@ -75,8 +67,7 @@ const selector = createMemoizedSelector(
forEach(node.data.inputs, (field) => { forEach(node.data.inputs, (field) => {
const fieldTemplate = nodeTemplate.inputs[field.name]; const fieldTemplate = nodeTemplate.inputs[field.name];
const hasConnection = connectedEdges.some( const hasConnection = connectedEdges.some(
(edge) => (edge) => edge.target === node.id && edge.targetHandle === field.name
edge.target === node.id && edge.targetHandle === field.name
); );
if (!fieldTemplate) { if (!fieldTemplate) {
@ -84,11 +75,7 @@ const selector = createMemoizedSelector(
return; return;
} }
if ( if (fieldTemplate.required && field.value === undefined && !hasConnection) {
fieldTemplate.required &&
field.value === undefined &&
!hasConnection
) {
reasons.push( reasons.push(
i18n.t('parameters.invoke.missingInputForField', { i18n.t('parameters.invoke.missingInputForField', {
nodeLabel: node.data.label || nodeTemplate.title, nodeLabel: node.data.label || nodeTemplate.title,
@ -101,10 +88,7 @@ const selector = createMemoizedSelector(
}); });
} }
} else { } else {
if ( if (dynamicPrompts.prompts.length === 0 && getShouldProcessPrompt(positivePrompt)) {
dynamicPrompts.prompts.length === 0 &&
getShouldProcessPrompt(positivePrompt)
) {
reasons.push(i18n.t('parameters.invoke.noPrompts')); reasons.push(i18n.t('parameters.invoke.noPrompts'));
} }
@ -134,9 +118,7 @@ const selector = createMemoizedSelector(
if ( if (
!ca.controlImage || !ca.controlImage ||
(isControlNetOrT2IAdapter(ca) && (isControlNetOrT2IAdapter(ca) && !ca.processedControlImage && ca.processorType !== 'none')
!ca.processedControlImage &&
ca.processorType !== 'none')
) { ) {
reasons.push( reasons.push(
i18n.t('parameters.invoke.noControlImageForControlAdapter', { i18n.t('parameters.invoke.noControlImageForControlAdapter', {

View File

@ -27,14 +27,7 @@ export const useModelCombobox = <T extends AnyModelConfigEntity>(
arg: UseModelComboboxArg<T> arg: UseModelComboboxArg<T>
): UseModelComboboxReturn => { ): UseModelComboboxReturn => {
const { t } = useTranslation(); const { t } = useTranslation();
const { const { modelEntities, selectedModel, getIsDisabled, onChange, isLoading, optionsFilter = () => true } = arg;
modelEntities,
selectedModel,
getIsDisabled,
onChange,
isLoading,
optionsFilter = () => true,
} = arg;
const options = useMemo<ComboboxOption[]>(() => { const options = useMemo<ComboboxOption[]>(() => {
if (!modelEntities) { if (!modelEntities) {
return []; return [];
@ -49,10 +42,7 @@ export const useModelCombobox = <T extends AnyModelConfigEntity>(
}, [optionsFilter, getIsDisabled, modelEntities]); }, [optionsFilter, getIsDisabled, modelEntities]);
const value = useMemo( const value = useMemo(
() => () => options.find((m) => (selectedModel ? m.value === getModelId(selectedModel) : false)),
options.find((m) =>
selectedModel ? m.value === getModelId(selectedModel) : false
),
[options, selectedModel] [options, selectedModel]
); );

View File

@ -20,12 +20,7 @@ export const areAnyPixelsBlack = (pixels: Uint8ClampedArray) => {
const len = pixels.length; const len = pixels.length;
let i = 0; let i = 0;
for (i; i < len; ) { for (i; i < len; ) {
if ( if (pixels[i++] === 0 && pixels[i++] === 0 && pixels[i++] === 0 && pixels[i++] === 255) {
pixels[i++] === 0 &&
pixels[i++] === 0 &&
pixels[i++] === 0 &&
pixels[i++] === 255
) {
return true; return true;
} }
} }

View File

@ -1,2 +1 @@
export const colorTokenToCssVar = (colorToken: string) => export const colorTokenToCssVar = (colorToken: string) => `var(--invoke-colors-${colorToken.split('.').join('-')})`;
`var(--invoke-colors-${colorToken.split('.').join('-')})`;

View File

@ -8,12 +8,7 @@ export type GenerateSeedsArg = {
max?: number; max?: number;
}; };
export const generateSeeds = ({ export const generateSeeds = ({ count, start, min = NUMPY_RAND_MIN, max = NUMPY_RAND_MAX }: GenerateSeedsArg) => {
count,
start,
min = NUMPY_RAND_MIN,
max = NUMPY_RAND_MAX,
}: GenerateSeedsArg) => {
const first = start ?? random(min, max); const first = start ?? random(min, max);
const seeds: number[] = []; const seeds: number[] = [];
for (let i = first; i < first + count; i++) { for (let i = first; i < first + count; i++) {
@ -22,7 +17,4 @@ export const generateSeeds = ({
return seeds; return seeds;
}; };
export const generateOneSeed = ( export const generateOneSeed = (min: number = NUMPY_RAND_MIN, max: number = NUMPY_RAND_MAX) => random(min, max);
min: number = NUMPY_RAND_MIN,
max: number = NUMPY_RAND_MAX
) => random(min, max);

View File

@ -1,10 +1,7 @@
export const roundDownToMultiple = (num: number, multiple: number): number => { export const roundDownToMultiple = (num: number, multiple: number): number => {
return Math.floor(num / multiple) * multiple; return Math.floor(num / multiple) * multiple;
}; };
export const roundDownToMultipleMin = ( export const roundDownToMultipleMin = (num: number, multiple: number): number => {
num: number,
multiple: number
): number => {
return Math.max(multiple, Math.floor(num / multiple) * multiple); return Math.max(multiple, Math.floor(num / multiple) * multiple);
}; };

View File

@ -1,8 +1,4 @@
import { import { Button, ConfirmationAlertDialog, useDisclosure } from '@invoke-ai/ui-library';
Button,
ConfirmationAlertDialog,
useDisclosure,
} from '@invoke-ai/ui-library';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { clearCanvasHistory } from 'features/canvas/store/canvasSlice'; import { clearCanvasHistory } from 'features/canvas/store/canvasSlice';
@ -15,19 +11,11 @@ const ClearCanvasHistoryButtonModal = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { t } = useTranslation(); const { t } = useTranslation();
const { isOpen, onOpen, onClose } = useDisclosure(); const { isOpen, onOpen, onClose } = useDisclosure();
const acceptCallback = useCallback( const acceptCallback = useCallback(() => dispatch(clearCanvasHistory()), [dispatch]);
() => dispatch(clearCanvasHistory()),
[dispatch]
);
return ( return (
<> <>
<Button <Button onClick={onOpen} size="sm" leftIcon={<PiTrashSimpleFill />} isDisabled={isStaging}>
onClick={onOpen}
size="sm"
leftIcon={<PiTrashSimpleFill />}
isDisabled={isStaging}
>
{t('unifiedCanvas.clearCanvasHistory')} {t('unifiedCanvas.clearCanvasHistory')}
</Button> </Button>
<ConfirmationAlertDialog <ConfirmationAlertDialog

View File

@ -19,10 +19,7 @@ import {
$tool, $tool,
} from 'features/canvas/store/canvasNanostore'; } from 'features/canvas/store/canvasNanostore';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { import { canvasResized, selectCanvasSlice } from 'features/canvas/store/canvasSlice';
canvasResized,
selectCanvasSlice,
} from 'features/canvas/store/canvasSlice';
import type Konva from 'konva'; import type Konva from 'konva';
import type { KonvaEventObject } from 'konva/lib/Node'; import type { KonvaEventObject } from 'konva/lib/Node';
import type { Vector2d } from 'konva/lib/types'; import type { Vector2d } from 'konva/lib/types';
@ -55,18 +52,12 @@ const ChakraStage = chakra(Stage, {
const IAICanvas = () => { const IAICanvas = () => {
const isStaging = useAppSelector(isStagingSelector); const isStaging = useAppSelector(isStagingSelector);
const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled); const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled);
const shouldShowBoundingBox = useAppSelector( const shouldShowBoundingBox = useAppSelector((s) => s.canvas.shouldShowBoundingBox);
(s) => s.canvas.shouldShowBoundingBox
);
const shouldShowGrid = useAppSelector((s) => s.canvas.shouldShowGrid); const shouldShowGrid = useAppSelector((s) => s.canvas.shouldShowGrid);
const stageScale = useAppSelector((s) => s.canvas.stageScale); const stageScale = useAppSelector((s) => s.canvas.stageScale);
const shouldShowIntermediates = useAppSelector( const shouldShowIntermediates = useAppSelector((s) => s.canvas.shouldShowIntermediates);
(s) => s.canvas.shouldShowIntermediates
);
const shouldAntialias = useAppSelector((s) => s.canvas.shouldAntialias); const shouldAntialias = useAppSelector((s) => s.canvas.shouldAntialias);
const shouldRestrictStrokesToBox = useAppSelector( const shouldRestrictStrokesToBox = useAppSelector((s) => s.canvas.shouldRestrictStrokesToBox);
(s) => s.canvas.shouldRestrictStrokesToBox
);
const { stageCoordinates, stageDimensions } = useAppSelector(selector); const { stageCoordinates, stageDimensions } = useAppSelector(selector);
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const containerRef = useRef<HTMLDivElement>(null); const containerRef = useRef<HTMLDivElement>(null);
@ -95,22 +86,12 @@ const IAICanvas = () => {
return 'default'; return 'default';
} }
return 'none'; return 'none';
}, [ }, [isMouseOverBoundingBox, isMovingStage, isStaging, isTransformingBoundingBox, shouldRestrictStrokesToBox, tool]);
isMouseOverBoundingBox,
isMovingStage,
isStaging,
isTransformingBoundingBox,
shouldRestrictStrokesToBox,
tool,
]);
const canvasBaseLayerRefCallback = useCallback( const canvasBaseLayerRefCallback = useCallback((layerElement: Konva.Layer) => {
(layerElement: Konva.Layer) => { $canvasBaseLayer.set(layerElement);
$canvasBaseLayer.set(layerElement); canvasBaseLayerRef.current = layerElement;
canvasBaseLayerRef.current = layerElement; }, []);
},
[]
);
const lastCursorPositionRef = useRef<Vector2d>({ x: 0, y: 0 }); const lastCursorPositionRef = useRef<Vector2d>({ x: 0, y: 0 });
@ -120,18 +101,10 @@ const IAICanvas = () => {
const handleWheel = useCanvasWheel(stageRef); const handleWheel = useCanvasWheel(stageRef);
const handleMouseDown = useCanvasMouseDown(stageRef); const handleMouseDown = useCanvasMouseDown(stageRef);
const handleMouseUp = useCanvasMouseUp(stageRef, didMouseMoveRef); const handleMouseUp = useCanvasMouseUp(stageRef, didMouseMoveRef);
const handleMouseMove = useCanvasMouseMove( const handleMouseMove = useCanvasMouseMove(stageRef, didMouseMoveRef, lastCursorPositionRef);
stageRef, const { handleDragStart, handleDragMove, handleDragEnd } = useCanvasDragMove();
didMouseMoveRef,
lastCursorPositionRef
);
const { handleDragStart, handleDragMove, handleDragEnd } =
useCanvasDragMove();
const handleMouseOut = useCanvasMouseOut(); const handleMouseOut = useCanvasMouseOut();
const handleContextMenu = useCallback( const handleContextMenu = useCallback((e: KonvaEventObject<MouseEvent>) => e.evt.preventDefault(), []);
(e: KonvaEventObject<MouseEvent>) => e.evt.preventDefault(),
[]
);
useEffect(() => { useEffect(() => {
if (!containerRef.current) { if (!containerRef.current) {
@ -169,14 +142,7 @@ const IAICanvas = () => {
const scale = useMemo(() => ({ x: stageScale, y: stageScale }), [stageScale]); const scale = useMemo(() => ({ x: stageScale, y: stageScale }), [stageScale]);
return ( return (
<Flex <Flex id="canvas-container" ref={containerRef} position="relative" height="100%" width="100%" borderRadius="base">
id="canvas-container"
ref={containerRef}
position="relative"
height="100%"
width="100%"
borderRadius="base"
>
<Box position="absolute"> <Box position="absolute">
<ChakraStage <ChakraStage
tabIndex={-1} tabIndex={-1}
@ -205,19 +171,10 @@ const IAICanvas = () => {
<IAICanvasGrid /> <IAICanvasGrid />
</Layer> </Layer>
<Layer <Layer id="base" ref={canvasBaseLayerRefCallback} listening={false} imageSmoothingEnabled={shouldAntialias}>
id="base"
ref={canvasBaseLayerRefCallback}
listening={false}
imageSmoothingEnabled={shouldAntialias}
>
<IAICanvasObjectRenderer /> <IAICanvasObjectRenderer />
</Layer> </Layer>
<Layer <Layer id="mask" visible={isMaskEnabled && !isStaging} listening={false}>
id="mask"
visible={isMaskEnabled && !isStaging}
listening={false}
>
<IAICanvasMaskLines visible={true} listening={false} /> <IAICanvasMaskLines visible={true} listening={false} />
<IAICanvasMaskCompositer listening={false} /> <IAICanvasMaskCompositer listening={false} />
</Layer> </Layer>
@ -225,17 +182,10 @@ const IAICanvas = () => {
<IAICanvasBoundingBoxOverlay /> <IAICanvasBoundingBoxOverlay />
</Layer> </Layer>
<Layer id="preview" imageSmoothingEnabled={shouldAntialias}> <Layer id="preview" imageSmoothingEnabled={shouldAntialias}>
{!isStaging && ( {!isStaging && <IAICanvasToolPreview visible={tool !== 'move'} listening={false} />}
<IAICanvasToolPreview
visible={tool !== 'move'}
listening={false}
/>
)}
<IAICanvasStagingArea listening={false} visible={isStaging} /> <IAICanvasStagingArea listening={false} visible={isStaging} />
{shouldShowIntermediates && <IAICanvasIntermediateImage />} {shouldShowIntermediates && <IAICanvasIntermediateImage />}
<IAICanvasBoundingBox <IAICanvasBoundingBox visible={shouldShowBoundingBox && !isStaging} />
visible={shouldShowBoundingBox && !isStaging}
/>
</Layer> </Layer>
</ChakraStage> </ChakraStage>
</Box> </Box>

View File

@ -5,12 +5,7 @@ import { memo } from 'react';
import { Group, Rect } from 'react-konva'; import { Group, Rect } from 'react-konva';
const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => { const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
const { const { boundingBoxCoordinates, boundingBoxDimensions, stageDimensions, stageCoordinates } = canvas;
boundingBoxCoordinates,
boundingBoxDimensions,
stageDimensions,
stageCoordinates,
} = canvas;
return { return {
boundingBoxCoordinates, boundingBoxCoordinates,
@ -21,15 +16,8 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
}); });
const IAICanvasBoundingBoxOverlay = () => { const IAICanvasBoundingBoxOverlay = () => {
const { const { boundingBoxCoordinates, boundingBoxDimensions, stageCoordinates, stageDimensions } = useAppSelector(selector);
boundingBoxCoordinates, const shouldDarkenOutsideBoundingBox = useAppSelector((s) => s.canvas.shouldDarkenOutsideBoundingBox);
boundingBoxDimensions,
stageCoordinates,
stageDimensions,
} = useAppSelector(selector);
const shouldDarkenOutsideBoundingBox = useAppSelector(
(s) => s.canvas.shouldDarkenOutsideBoundingBox
);
const stageScale = useAppSelector((s) => s.canvas.stageScale); const stageScale = useAppSelector((s) => s.canvas.stageScale);
return ( return (

View File

@ -13,13 +13,8 @@ type IAICanvasImageProps = {
}; };
const IAICanvasImage = (props: IAICanvasImageProps) => { const IAICanvasImage = (props: IAICanvasImageProps) => {
const { x, y, imageName } = props.canvasImage; const { x, y, imageName } = props.canvasImage;
const { currentData: imageDTO, isError } = useGetImageDTOQuery( const { currentData: imageDTO, isError } = useGetImageDTOQuery(imageName ?? skipToken);
imageName ?? skipToken const [image, status] = useImage(imageDTO?.image_url ?? '', $authToken.get() ? 'use-credentials' : 'anonymous');
);
const [image, status] = useImage(
imageDTO?.image_url ?? '',
$authToken.get() ? 'use-credentials' : 'anonymous'
);
if (isError || status === 'failed') { if (isError || status === 'failed') {
return <IAICanvasImageErrorFallback canvasImage={props.canvasImage} />; return <IAICanvasImageErrorFallback canvasImage={props.canvasImage} />;

View File

@ -7,9 +7,7 @@ import { Group, Rect, Text } from 'react-konva';
type IAICanvasImageErrorFallbackProps = { type IAICanvasImageErrorFallbackProps = {
canvasImage: CanvasImage; canvasImage: CanvasImage;
}; };
const IAICanvasImageErrorFallback = ({ const IAICanvasImageErrorFallback = ({ canvasImage }: IAICanvasImageErrorFallbackProps) => {
canvasImage,
}: IAICanvasImageErrorFallbackProps) => {
const [rectFill, textFill] = useToken('colors', ['base.500', 'base.900']); const [rectFill, textFill] = useToken('colors', ['base.500', 'base.900']);
const { t } = useTranslation(); const { t } = useTranslation();
return ( return (

View File

@ -5,26 +5,20 @@ import { selectSystemSlice } from 'features/system/store/systemSlice';
import { memo, useEffect, useState } from 'react'; import { memo, useEffect, useState } from 'react';
import { Image as KonvaImage } from 'react-konva'; import { Image as KonvaImage } from 'react-konva';
const progressImageSelector = createMemoizedSelector( const progressImageSelector = createMemoizedSelector([selectSystemSlice, selectCanvasSlice], (system, canvas) => {
[selectSystemSlice, selectCanvasSlice], const { denoiseProgress } = system;
(system, canvas) => { const { batchIds } = canvas;
const { denoiseProgress } = system;
const { batchIds } = canvas;
return { return {
progressImage: progressImage:
denoiseProgress && batchIds.includes(denoiseProgress.batch_id) denoiseProgress && batchIds.includes(denoiseProgress.batch_id) ? denoiseProgress.progress_image : undefined,
? denoiseProgress.progress_image boundingBox: canvas.layerState.stagingArea.boundingBox,
: undefined, };
boundingBox: canvas.layerState.stagingArea.boundingBox, });
};
}
);
const IAICanvasIntermediateImage = () => { const IAICanvasIntermediateImage = () => {
const { progressImage, boundingBox } = useAppSelector(progressImageSelector); const { progressImage, boundingBox } = useAppSelector(progressImageSelector);
const [loadedImageElement, setLoadedImageElement] = const [loadedImageElement, setLoadedImageElement] = useState<HTMLImageElement | null>(null);
useState<HTMLImageElement | null>(null);
useEffect(() => { useEffect(() => {
if (!progressImage) { if (!progressImage) {

View File

@ -9,30 +9,22 @@ import { isNumber } from 'lodash-es';
import { memo, useCallback, useEffect, useMemo, useRef, useState } from 'react'; import { memo, useCallback, useEffect, useMemo, useRef, useState } from 'react';
import { Rect } from 'react-konva'; import { Rect } from 'react-konva';
export const canvasMaskCompositerSelector = createMemoizedSelector( export const canvasMaskCompositerSelector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
selectCanvasSlice, return {
(canvas) => { stageCoordinates: canvas.stageCoordinates,
return { stageDimensions: canvas.stageDimensions,
stageCoordinates: canvas.stageCoordinates, };
stageDimensions: canvas.stageDimensions, });
};
}
);
type IAICanvasMaskCompositerProps = RectConfig; type IAICanvasMaskCompositerProps = RectConfig;
const IAICanvasMaskCompositer = (props: IAICanvasMaskCompositerProps) => { const IAICanvasMaskCompositer = (props: IAICanvasMaskCompositerProps) => {
const { ...rest } = props; const { ...rest } = props;
const { stageCoordinates, stageDimensions } = useAppSelector( const { stageCoordinates, stageDimensions } = useAppSelector(canvasMaskCompositerSelector);
canvasMaskCompositerSelector
);
const stageScale = useAppSelector((s) => s.canvas.stageScale); const stageScale = useAppSelector((s) => s.canvas.stageScale);
const maskColorString = useAppSelector((s) => const maskColorString = useAppSelector((s) => rgbaColorToString(s.canvas.maskColor));
rgbaColorToString(s.canvas.maskColor) const [fillPatternImage, setFillPatternImage] = useState<HTMLImageElement | null>(null);
);
const [fillPatternImage, setFillPatternImage] =
useState<HTMLImageElement | null>(null);
const [offset, setOffset] = useState<number>(0); const [offset, setOffset] = useState<number>(0);
@ -66,10 +58,7 @@ const IAICanvasMaskCompositer = (props: IAICanvasMaskCompositerProps) => {
return () => clearInterval(timer); return () => clearInterval(timer);
}, []); }, []);
const fillPatternScale = useMemo( const fillPatternScale = useMemo(() => ({ x: 1 / stageScale, y: 1 / stageScale }), [stageScale]);
() => ({ x: 1 / stageScale, y: 1 / stageScale }),
[stageScale]
);
if ( if (
!fillPatternImage || !fillPatternImage ||

View File

@ -27,9 +27,7 @@ const IAICanvasLines = (props: InpaintingCanvasLinesProps) => {
lineJoin="round" lineJoin="round"
shadowForStrokeEnabled={false} shadowForStrokeEnabled={false}
listening={false} listening={false}
globalCompositeOperation={ globalCompositeOperation={line.tool === 'brush' ? 'source-over' : 'destination-out'}
line.tool === 'brush' ? 'source-over' : 'destination-out'
}
/> />
))} ))}
</Group> </Group>

View File

@ -31,9 +31,7 @@ const IAICanvasObjectRenderer = () => {
lineJoin="round" lineJoin="round"
shadowForStrokeEnabled={false} shadowForStrokeEnabled={false}
listening={false} listening={false}
globalCompositeOperation={ globalCompositeOperation={obj.tool === 'brush' ? 'source-over' : 'destination-out'}
obj.tool === 'brush' ? 'source-over' : 'destination-out'
}
/> />
); );
if (obj.clip) { if (obj.clip) {

View File

@ -22,9 +22,7 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
return { return {
currentStagingAreaImage: currentStagingAreaImage:
images.length > 0 && selectedImageIndex !== undefined images.length > 0 && selectedImageIndex !== undefined ? images[selectedImageIndex] : undefined,
? images[selectedImageIndex]
: undefined,
isOnFirstImage: selectedImageIndex === 0, isOnFirstImage: selectedImageIndex === 0,
isOnLastImage: selectedImageIndex === images.length - 1, isOnLastImage: selectedImageIndex === images.length - 1,
shouldShowStagingImage, shouldShowStagingImage,
@ -39,21 +37,12 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
type Props = GroupConfig; type Props = GroupConfig;
const IAICanvasStagingArea = (props: Props) => { const IAICanvasStagingArea = (props: Props) => {
const { const { currentStagingAreaImage, shouldShowStagingImage, shouldShowStagingOutline, x, y, width, height } =
currentStagingAreaImage, useAppSelector(selector);
shouldShowStagingImage,
shouldShowStagingOutline,
x,
y,
width,
height,
} = useAppSelector(selector);
return ( return (
<Group {...props}> <Group {...props}>
{shouldShowStagingImage && currentStagingAreaImage && ( {shouldShowStagingImage && currentStagingAreaImage && <IAICanvasImage canvasImage={currentStagingAreaImage} />}
<IAICanvasImage canvasImage={currentStagingAreaImage} />
)}
{shouldShowStagingOutline && ( {shouldShowStagingOutline && (
<Group listening={false}> <Group listening={false}>
<Rect <Rect

View File

@ -38,8 +38,7 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
return { return {
currentIndex: selectedImageIndex, currentIndex: selectedImageIndex,
total: images.length, total: images.length,
currentStagingAreaImage: currentStagingAreaImage: images.length > 0 ? images[selectedImageIndex] : undefined,
images.length > 0 ? images[selectedImageIndex] : undefined,
shouldShowStagingImage, shouldShowStagingImage,
shouldShowStagingOutline, shouldShowStagingOutline,
}; };
@ -47,12 +46,7 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
const IAICanvasStagingAreaToolbar = () => { const IAICanvasStagingAreaToolbar = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { const { currentStagingAreaImage, shouldShowStagingImage, currentIndex, total } = useAppSelector(selector);
currentStagingAreaImage,
shouldShowStagingImage,
currentIndex,
total,
} = useAppSelector(selector);
const { t } = useTranslation(); const { t } = useTranslation();
@ -64,20 +58,11 @@ const IAICanvasStagingAreaToolbar = () => {
dispatch(setShouldShowStagingOutline(false)); dispatch(setShouldShowStagingOutline(false));
}, [dispatch]); }, [dispatch]);
const handlePrevImage = useCallback( const handlePrevImage = useCallback(() => dispatch(prevStagingAreaImage()), [dispatch]);
() => dispatch(prevStagingAreaImage()),
[dispatch]
);
const handleNextImage = useCallback( const handleNextImage = useCallback(() => dispatch(nextStagingAreaImage()), [dispatch]);
() => dispatch(nextStagingAreaImage()),
[dispatch]
);
const handleAccept = useCallback( const handleAccept = useCallback(() => dispatch(commitStagingAreaImage()), [dispatch]);
() => dispatch(commitStagingAreaImage()),
[dispatch]
);
useHotkeys(['left'], handlePrevImage, { useHotkeys(['left'], handlePrevImage, {
enabled: () => true, enabled: () => true,
@ -94,9 +79,7 @@ const IAICanvasStagingAreaToolbar = () => {
preventDefault: true, preventDefault: true,
}); });
const { data: imageDTO } = useGetImageDTOQuery( const { data: imageDTO } = useGetImageDTOQuery(currentStagingAreaImage?.imageName ?? skipToken);
currentStagingAreaImage?.imageName ?? skipToken
);
const handleToggleShouldShowStagingImage = useCallback(() => { const handleToggleShouldShowStagingImage = useCallback(() => {
dispatch(setShouldShowStagingImage(!shouldShowStagingImage)); dispatch(setShouldShowStagingImage(!shouldShowStagingImage));
@ -166,16 +149,8 @@ const IAICanvasStagingAreaToolbar = () => {
colorScheme="invokeBlue" colorScheme="invokeBlue"
/> />
<IconButton <IconButton
tooltip={ tooltip={shouldShowStagingImage ? t('unifiedCanvas.showResultsOn') : t('unifiedCanvas.showResultsOff')}
shouldShowStagingImage aria-label={shouldShowStagingImage ? t('unifiedCanvas.showResultsOn') : t('unifiedCanvas.showResultsOff')}
? t('unifiedCanvas.showResultsOn')
: t('unifiedCanvas.showResultsOff')
}
aria-label={
shouldShowStagingImage
? t('unifiedCanvas.showResultsOn')
: t('unifiedCanvas.showResultsOff')
}
data-alert={!shouldShowStagingImage} data-alert={!shouldShowStagingImage}
icon={shouldShowStagingImage ? <PiEyeBold /> : <PiEyeSlashBold />} icon={shouldShowStagingImage ? <PiEyeBold /> : <PiEyeSlashBold />}
onClick={handleToggleShouldShowStagingImage} onClick={handleToggleShouldShowStagingImage}

View File

@ -16,10 +16,7 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
stageDimensions: { width: stageWidth, height: stageHeight }, stageDimensions: { width: stageWidth, height: stageHeight },
stageCoordinates: { x: stageX, y: stageY }, stageCoordinates: { x: stageX, y: stageY },
boundingBoxDimensions: { width: boxWidth, height: boxHeight }, boundingBoxDimensions: { width: boxWidth, height: boxHeight },
scaledBoundingBoxDimensions: { scaledBoundingBoxDimensions: { width: scaledBoxWidth, height: scaledBoxHeight },
width: scaledBoxWidth,
height: scaledBoxHeight,
},
boundingBoxCoordinates: { x: boxX, y: boxY }, boundingBoxCoordinates: { x: boxX, y: boxY },
stageScale, stageScale,
shouldShowCanvasDebugInfo, shouldShowCanvasDebugInfo,
@ -31,10 +28,8 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
let boundingBoxColor = 'inherit'; let boundingBoxColor = 'inherit';
if ( if (
(boundingBoxScaleMethod === 'none' && (boundingBoxScaleMethod === 'none' && (boxWidth < 512 || boxHeight < 512)) ||
(boxWidth < 512 || boxHeight < 512)) || (boundingBoxScaleMethod === 'manual' && scaledBoxWidth * scaledBoxHeight < 512 * 512)
(boundingBoxScaleMethod === 'manual' &&
scaledBoxWidth * scaledBoxHeight < 512 * 512)
) { ) {
boundingBoxColor = warningColor; boundingBoxColor = warningColor;
} }
@ -45,14 +40,10 @@ const selector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
activeLayerColor, activeLayerColor,
layer, layer,
boundingBoxColor, boundingBoxColor,
boundingBoxCoordinatesString: `(${roundToHundreth(boxX)}, ${roundToHundreth( boundingBoxCoordinatesString: `(${roundToHundreth(boxX)}, ${roundToHundreth(boxY)})`,
boxY
)})`,
boundingBoxDimensionsString: `${boxWidth}×${boxHeight}`, boundingBoxDimensionsString: `${boxWidth}×${boxHeight}`,
scaledBoundingBoxDimensionsString: `${scaledBoxWidth}×${scaledBoxHeight}`, scaledBoundingBoxDimensionsString: `${scaledBoxWidth}×${scaledBoxHeight}`,
canvasCoordinatesString: `${roundToHundreth(stageX)}×${roundToHundreth( canvasCoordinatesString: `${roundToHundreth(stageX)}×${roundToHundreth(stageY)}`,
stageY
)}`,
canvasDimensionsString: `${stageWidth}×${stageHeight}`, canvasDimensionsString: `${stageWidth}×${stageHeight}`,
canvasScaleString: Math.round(stageScale * 100), canvasScaleString: Math.round(stageScale * 100),
shouldShowCanvasDebugInfo, shouldShowCanvasDebugInfo,
@ -99,9 +90,7 @@ const IAICanvasStatusText = () => {
bg="base.800" bg="base.800"
> >
<GenerationModeStatusText /> <GenerationModeStatusText />
<Box color={activeLayerColor}>{`${t('unifiedCanvas.activeLayer')}: ${t( <Box color={activeLayerColor}>{`${t('unifiedCanvas.activeLayer')}: ${t(`unifiedCanvas.${layer}`)}`}</Box>
`unifiedCanvas.${layer}`
)}`}</Box>
<Box>{`${t('unifiedCanvas.canvasScale')}: ${canvasScaleString}%`}</Box> <Box>{`${t('unifiedCanvas.canvasScale')}: ${canvasScaleString}%`}</Box>
{shouldPreserveMaskedArea && ( {shouldPreserveMaskedArea && (
<Box color={warningColor}> <Box color={warningColor}>
@ -109,9 +98,7 @@ const IAICanvasStatusText = () => {
</Box> </Box>
)} )}
{shouldShowBoundingBox && ( {shouldShowBoundingBox && (
<Box color={boundingBoxColor}>{`${t( <Box color={boundingBoxColor}>{`${t('unifiedCanvas.boundingBox')}: ${boundingBoxDimensionsString}`}</Box>
'unifiedCanvas.boundingBox'
)}: ${boundingBoxDimensionsString}`}</Box>
)} )}
{shouldShowScaledBoundingBox && ( {shouldShowScaledBoundingBox && (
<Box color={boundingBoxColor}>{`${t( <Box color={boundingBoxColor}>{`${t(
@ -120,15 +107,9 @@ const IAICanvasStatusText = () => {
)} )}
{shouldShowCanvasDebugInfo && ( {shouldShowCanvasDebugInfo && (
<> <>
<Box>{`${t( <Box>{`${t('unifiedCanvas.boundingBoxPosition')}: ${boundingBoxCoordinatesString}`}</Box>
'unifiedCanvas.boundingBoxPosition' <Box>{`${t('unifiedCanvas.canvasDimensions')}: ${canvasDimensionsString}`}</Box>
)}: ${boundingBoxCoordinatesString}`}</Box> <Box>{`${t('unifiedCanvas.canvasPosition')}: ${canvasCoordinatesString}`}</Box>
<Box>{`${t(
'unifiedCanvas.canvasDimensions'
)}: ${canvasDimensionsString}`}</Box>
<Box>{`${t(
'unifiedCanvas.canvasPosition'
)}: ${canvasCoordinatesString}`}</Box>
<IAICanvasStatusTextCursorPos /> <IAICanvasStatusTextCursorPos />
</> </>
)} )}

View File

@ -14,11 +14,7 @@ const IAICanvasStatusTextCursorPos = () => {
return `(${roundToHundreth(x)}, ${roundToHundreth(y)})`; return `(${roundToHundreth(x)}, ${roundToHundreth(y)})`;
}, [cursorPosition?.x, cursorPosition?.y]); }, [cursorPosition?.x, cursorPosition?.y]);
return ( return <Box>{`${t('unifiedCanvas.cursorPosition')}: ${cursorCoordinatesString}`}</Box>;
<Box>{`${t(
'unifiedCanvas.cursorPosition'
)}: ${cursorCoordinatesString}`}</Box>
);
}; };
export default memo(IAICanvasStatusTextCursorPos); export default memo(IAICanvasStatusTextCursorPos);

View File

@ -9,104 +9,84 @@ import {
} from 'features/canvas/store/canvasNanostore'; } from 'features/canvas/store/canvasNanostore';
import { selectCanvasSlice } from 'features/canvas/store/canvasSlice'; import { selectCanvasSlice } from 'features/canvas/store/canvasSlice';
import { rgbaColorToString } from 'features/canvas/util/colorToString'; import { rgbaColorToString } from 'features/canvas/util/colorToString';
import { import { COLOR_PICKER_SIZE, COLOR_PICKER_STROKE_RADIUS } from 'features/canvas/util/constants';
COLOR_PICKER_SIZE,
COLOR_PICKER_STROKE_RADIUS,
} from 'features/canvas/util/constants';
import type { GroupConfig } from 'konva/lib/Group'; import type { GroupConfig } from 'konva/lib/Group';
import { memo, useMemo } from 'react'; import { memo, useMemo } from 'react';
import { Circle, Group } from 'react-konva'; import { Circle, Group } from 'react-konva';
const canvasBrushPreviewSelector = createMemoizedSelector( const canvasBrushPreviewSelector = createMemoizedSelector(selectCanvasSlice, (canvas) => {
selectCanvasSlice, const { stageDimensions, boundingBoxCoordinates, boundingBoxDimensions, shouldRestrictStrokesToBox } = canvas;
(canvas) => {
const {
stageDimensions,
boundingBoxCoordinates,
boundingBoxDimensions,
shouldRestrictStrokesToBox,
} = canvas;
const clip = shouldRestrictStrokesToBox const clip = shouldRestrictStrokesToBox
? { ? {
clipX: boundingBoxCoordinates.x, clipX: boundingBoxCoordinates.x,
clipY: boundingBoxCoordinates.y, clipY: boundingBoxCoordinates.y,
clipWidth: boundingBoxDimensions.width, clipWidth: boundingBoxDimensions.width,
clipHeight: boundingBoxDimensions.height, clipHeight: boundingBoxDimensions.height,
} }
: {}; : {};
// // big brain time; this is the *inverse* of the clip that is needed for shouldRestrictStrokesToBox // // big brain time; this is the *inverse* of the clip that is needed for shouldRestrictStrokesToBox
// // it took some fiddling to work out, so I am leaving it here in case it is needed for something else... // // it took some fiddling to work out, so I am leaving it here in case it is needed for something else...
// const clipFunc = shouldRestrictStrokesToBox // const clipFunc = shouldRestrictStrokesToBox
// ? (ctx: SceneContext) => { // ? (ctx: SceneContext) => {
// console.log( // console.log(
// stageCoordinates.x / stageScale, // stageCoordinates.x / stageScale,
// stageCoordinates.y / stageScale, // stageCoordinates.y / stageScale,
// stageDimensions.height / stageScale, // stageDimensions.height / stageScale,
// stageDimensions.width / stageScale // stageDimensions.width / stageScale
// ); // );
// ctx.fillStyle = 'red'; // ctx.fillStyle = 'red';
// ctx.rect( // ctx.rect(
// -stageCoordinates.x / stageScale, // -stageCoordinates.x / stageScale,
// -stageCoordinates.y / stageScale, // -stageCoordinates.y / stageScale,
// stageDimensions.width / stageScale, // stageDimensions.width / stageScale,
// stageCoordinates.y / stageScale + boundingBoxCoordinates.y // stageCoordinates.y / stageScale + boundingBoxCoordinates.y
// ); // );
// ctx.rect( // ctx.rect(
// -stageCoordinates.x / stageScale, // -stageCoordinates.x / stageScale,
// boundingBoxCoordinates.y + boundingBoxDimensions.height, // boundingBoxCoordinates.y + boundingBoxDimensions.height,
// stageDimensions.width / stageScale, // stageDimensions.width / stageScale,
// stageDimensions.height / stageScale // stageDimensions.height / stageScale
// ); // );
// ctx.rect( // ctx.rect(
// -stageCoordinates.x / stageScale, // -stageCoordinates.x / stageScale,
// -stageCoordinates.y / stageScale, // -stageCoordinates.y / stageScale,
// stageCoordinates.x / stageScale + boundingBoxCoordinates.x, // stageCoordinates.x / stageScale + boundingBoxCoordinates.x,
// stageDimensions.height / stageScale // stageDimensions.height / stageScale
// ); // );
// ctx.rect( // ctx.rect(
// boundingBoxCoordinates.x + boundingBoxDimensions.width, // boundingBoxCoordinates.x + boundingBoxDimensions.width,
// -stageCoordinates.y / stageScale, // -stageCoordinates.y / stageScale,
// stageDimensions.width / stageScale - // stageDimensions.width / stageScale -
// (boundingBoxCoordinates.x + boundingBoxDimensions.width), // (boundingBoxCoordinates.x + boundingBoxDimensions.width),
// stageDimensions.height / stageScale // stageDimensions.height / stageScale
// ); // );
// } // }
// : undefined; // : undefined;
return { return {
clip, clip,
stageDimensions, stageDimensions,
}; };
} });
);
/** /**
* Draws a black circle around the canvas brush preview. * Draws a black circle around the canvas brush preview.
*/ */
const IAICanvasToolPreview = (props: GroupConfig) => { const IAICanvasToolPreview = (props: GroupConfig) => {
const radius = useAppSelector((s) => s.canvas.brushSize / 2); const radius = useAppSelector((s) => s.canvas.brushSize / 2);
const maskColorString = useAppSelector((s) => const maskColorString = useAppSelector((s) => rgbaColorToString({ ...s.canvas.maskColor, a: 0.5 }));
rgbaColorToString({ ...s.canvas.maskColor, a: 0.5 })
);
const tool = useStore($tool); const tool = useStore($tool);
const layer = useAppSelector((s) => s.canvas.layer); const layer = useAppSelector((s) => s.canvas.layer);
const dotRadius = useAppSelector((s) => 1.5 / s.canvas.stageScale); const dotRadius = useAppSelector((s) => 1.5 / s.canvas.stageScale);
const strokeWidth = useAppSelector((s) => 1.5 / s.canvas.stageScale); const strokeWidth = useAppSelector((s) => 1.5 / s.canvas.stageScale);
const brushColorString = useAppSelector((s) => const brushColorString = useAppSelector((s) => rgbaColorToString(s.canvas.brushColor));
rgbaColorToString(s.canvas.brushColor) const colorPickerColorString = useAppSelector((s) => rgbaColorToString(s.canvas.colorPickerColor));
);
const colorPickerColorString = useAppSelector((s) =>
rgbaColorToString(s.canvas.colorPickerColor)
);
const colorPickerInnerRadius = useAppSelector( const colorPickerInnerRadius = useAppSelector(
(s) => (s) => (COLOR_PICKER_SIZE - COLOR_PICKER_STROKE_RADIUS + 1) / s.canvas.stageScale
(COLOR_PICKER_SIZE - COLOR_PICKER_STROKE_RADIUS + 1) / s.canvas.stageScale
);
const colorPickerOuterRadius = useAppSelector(
(s) => COLOR_PICKER_SIZE / s.canvas.stageScale
); );
const colorPickerOuterRadius = useAppSelector((s) => COLOR_PICKER_SIZE / s.canvas.stageScale);
const { clip, stageDimensions } = useAppSelector(canvasBrushPreviewSelector); const { clip, stageDimensions } = useAppSelector(canvasBrushPreviewSelector);
const cursorPosition = useStore($cursorPosition); const cursorPosition = useStore($cursorPosition);
@ -123,8 +103,7 @@ const IAICanvasToolPreview = (props: GroupConfig) => {
); );
const shouldDrawBrushPreview = useMemo( const shouldDrawBrushPreview = useMemo(
() => () => !(isMovingBoundingBox || isTransformingBoundingBox || !cursorPosition),
!(isMovingBoundingBox || isTransformingBoundingBox || !cursorPosition),
[cursorPosition, isMovingBoundingBox, isTransformingBoundingBox] [cursorPosition, isMovingBoundingBox, isTransformingBoundingBox]
); );
@ -162,9 +141,7 @@ const IAICanvasToolPreview = (props: GroupConfig) => {
y={brushY} y={brushY}
radius={radius} radius={radius}
fill={layer === 'mask' ? maskColorString : brushColorString} fill={layer === 'mask' ? maskColorString : brushColorString}
globalCompositeOperation={ globalCompositeOperation={tool === 'eraser' ? 'destination-out' : 'source-out'}
tool === 'eraser' ? 'destination-out' : 'source-out'
}
listening={false} listening={false}
/> />
<Circle <Circle
@ -187,20 +164,8 @@ const IAICanvasToolPreview = (props: GroupConfig) => {
/> />
</> </>
)} )}
<Circle <Circle x={brushX} y={brushY} radius={dotRadius * 2} fill="rgba(255,255,255,0.4)" listening={false} />
x={brushX} <Circle x={brushX} y={brushY} radius={dotRadius} fill="rgba(0,0,0,1)" listening={false} />
y={brushY}
radius={dotRadius * 2}
fill="rgba(255,255,255,0.4)"
listening={false}
/>
<Circle
x={brushX}
y={brushY}
radius={dotRadius}
fill="rgba(0,0,0,1)"
listening={false}
/>
</Group> </Group>
); );
}; };

View File

@ -1,11 +1,7 @@
import { useShiftModifier } from '@invoke-ai/ui-library'; import { useShiftModifier } from '@invoke-ai/ui-library';
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { import { roundDownToMultiple, roundDownToMultipleMin, roundToMultiple } from 'common/util/roundDownToMultiple';
roundDownToMultiple,
roundDownToMultipleMin,
roundToMultiple,
} from 'common/util/roundDownToMultiple';
import { import {
$isDrawing, $isDrawing,
$isMouseOverBoundingBox, $isMouseOverBoundingBox,
@ -20,10 +16,7 @@ import {
setBoundingBoxDimensions, setBoundingBoxDimensions,
setShouldSnapToGrid, setShouldSnapToGrid,
} from 'features/canvas/store/canvasSlice'; } from 'features/canvas/store/canvasSlice';
import { import { CANVAS_GRID_SIZE_COARSE, CANVAS_GRID_SIZE_FINE } from 'features/canvas/store/constants';
CANVAS_GRID_SIZE_COARSE,
CANVAS_GRID_SIZE_FINE,
} from 'features/canvas/store/constants';
import { calculateNewSize } from 'features/parameters/components/ImageSize/calculateNewSize'; import { calculateNewSize } from 'features/parameters/components/ImageSize/calculateNewSize';
import { selectOptimalDimension } from 'features/parameters/store/generationSlice'; import { selectOptimalDimension } from 'features/parameters/store/generationSlice';
import type Konva from 'konva'; import type Konva from 'konva';
@ -41,12 +34,8 @@ type IAICanvasBoundingBoxPreviewProps = GroupConfig;
const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => { const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => {
const { ...rest } = props; const { ...rest } = props;
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const boundingBoxCoordinates = useAppSelector( const boundingBoxCoordinates = useAppSelector((s) => s.canvas.boundingBoxCoordinates);
(s) => s.canvas.boundingBoxCoordinates const boundingBoxDimensions = useAppSelector((s) => s.canvas.boundingBoxDimensions);
);
const boundingBoxDimensions = useAppSelector(
(s) => s.canvas.boundingBoxDimensions
);
const stageScale = useAppSelector((s) => s.canvas.stageScale); const stageScale = useAppSelector((s) => s.canvas.stageScale);
const shouldSnapToGrid = useAppSelector((s) => s.canvas.shouldSnapToGrid); const shouldSnapToGrid = useAppSelector((s) => s.canvas.shouldSnapToGrid);
const hitStrokeWidth = useAppSelector((s) => 20 / s.canvas.stageScale); const hitStrokeWidth = useAppSelector((s) => 20 / s.canvas.stageScale);
@ -59,9 +48,7 @@ const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => {
const isDrawing = useStore($isDrawing); const isDrawing = useStore($isDrawing);
const isMovingBoundingBox = useStore($isMovingBoundingBox); const isMovingBoundingBox = useStore($isMovingBoundingBox);
const isTransformingBoundingBox = useStore($isTransformingBoundingBox); const isTransformingBoundingBox = useStore($isTransformingBoundingBox);
const isMouseOverBoundingBoxOutline = useStore( const isMouseOverBoundingBoxOutline = useStore($isMouseOverBoundingBoxOutline);
$isMouseOverBoundingBoxOutline
);
useEffect(() => { useEffect(() => {
if (!transformerRef.current || !shapeRef.current) { if (!transformerRef.current || !shapeRef.current) {
@ -71,14 +58,8 @@ const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => {
transformerRef.current.getLayer()?.batchDraw(); transformerRef.current.getLayer()?.batchDraw();
}, []); }, []);
const gridSize = useMemo( const gridSize = useMemo(() => (shift ? CANVAS_GRID_SIZE_FINE : CANVAS_GRID_SIZE_COARSE), [shift]);
() => (shift ? CANVAS_GRID_SIZE_FINE : CANVAS_GRID_SIZE_COARSE), const scaledStep = useMemo(() => gridSize * stageScale, [gridSize, stageScale]);
[shift]
);
const scaledStep = useMemo(
() => gridSize * stageScale,
[gridSize, stageScale]
);
useHotkeys( useHotkeys(
'N', 'N',
@ -143,10 +124,7 @@ const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => {
const y = Math.round(rect.y()); const y = Math.round(rect.y());
if (aspectRatio.isLocked) { if (aspectRatio.isLocked) {
const newDimensions = calculateNewSize( const newDimensions = calculateNewSize(aspectRatio.value, width * height);
aspectRatio.value,
width * height
);
dispatch( dispatch(
setBoundingBoxDimensions( setBoundingBoxDimensions(
{ {
@ -186,14 +164,7 @@ const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => {
rect.scaleX(1); rect.scaleX(1);
rect.scaleY(1); rect.scaleY(1);
}, },
[ [aspectRatio.isLocked, aspectRatio.value, dispatch, shouldSnapToGrid, gridSize, optimalDimension]
aspectRatio.isLocked,
aspectRatio.value,
dispatch,
shouldSnapToGrid,
gridSize,
optimalDimension,
]
); );
const anchorDragBoundFunc = useCallback( const anchorDragBoundFunc = useCallback(
@ -255,9 +226,7 @@ const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => {
}, []); }, []);
const handleMouseOut = useCallback(() => { const handleMouseOut = useCallback(() => {
!isTransformingBoundingBox && !isTransformingBoundingBox && !isMovingBoundingBox && $isMouseOverBoundingBoxOutline.set(false);
!isMovingBoundingBox &&
$isMouseOverBoundingBoxOutline.set(false);
}, [isMovingBoundingBox, isTransformingBoundingBox]); }, [isMovingBoundingBox, isTransformingBoundingBox]);
const handleMouseEnterBoundingBox = useCallback(() => { const handleMouseEnterBoundingBox = useCallback(() => {
@ -269,35 +238,18 @@ const IAICanvasBoundingBox = (props: IAICanvasBoundingBoxPreviewProps) => {
}, []); }, []);
const stroke = useMemo(() => { const stroke = useMemo(() => {
if ( if (isMouseOverBoundingBoxOutline || isMovingBoundingBox || isTransformingBoundingBox) {
isMouseOverBoundingBoxOutline ||
isMovingBoundingBox ||
isTransformingBoundingBox
) {
return 'rgba(255,255,255,0.5)'; return 'rgba(255,255,255,0.5)';
} }
return 'white'; return 'white';
}, [ }, [isMouseOverBoundingBoxOutline, isMovingBoundingBox, isTransformingBoundingBox]);
isMouseOverBoundingBoxOutline,
isMovingBoundingBox,
isTransformingBoundingBox,
]);
const strokeWidth = useMemo(() => { const strokeWidth = useMemo(() => {
if ( if (isMouseOverBoundingBoxOutline || isMovingBoundingBox || isTransformingBoundingBox) {
isMouseOverBoundingBoxOutline ||
isMovingBoundingBox ||
isTransformingBoundingBox
) {
return 6 / stageScale; return 6 / stageScale;
} }
return 1 / stageScale; return 1 / stageScale;
}, [ }, [isMouseOverBoundingBoxOutline, isMovingBoundingBox, isTransformingBoundingBox, stageScale]);
isMouseOverBoundingBoxOutline,
isMovingBoundingBox,
isTransformingBoundingBox,
stageScale,
]);
const enabledAnchors = useMemo(() => { const enabledAnchors = useMemo(() => {
if (tool !== 'move') { if (tool !== 'move') {

View File

@ -30,11 +30,7 @@ import { memo, useCallback } from 'react';
import type { RgbaColor } from 'react-colorful'; import type { RgbaColor } from 'react-colorful';
import { useHotkeys } from 'react-hotkeys-hook'; import { useHotkeys } from 'react-hotkeys-hook';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { import { PiExcludeBold, PiFloppyDiskBackFill, PiTrashSimpleFill } from 'react-icons/pi';
PiExcludeBold,
PiFloppyDiskBackFill,
PiTrashSimpleFill,
} from 'react-icons/pi';
const formLabelProps: FormLabelProps = { const formLabelProps: FormLabelProps = {
flexGrow: 1, flexGrow: 1,
@ -46,9 +42,7 @@ const IAICanvasMaskOptions = () => {
const layer = useAppSelector((s) => s.canvas.layer); const layer = useAppSelector((s) => s.canvas.layer);
const maskColor = useAppSelector((s) => s.canvas.maskColor); const maskColor = useAppSelector((s) => s.canvas.maskColor);
const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled); const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled);
const shouldPreserveMaskedArea = useAppSelector( const shouldPreserveMaskedArea = useAppSelector((s) => s.canvas.shouldPreserveMaskedArea);
(s) => s.canvas.shouldPreserveMaskedArea
);
const isStaging = useAppSelector(isStagingSelector); const isStaging = useAppSelector(isStagingSelector);
useHotkeys( useHotkeys(
@ -134,38 +128,21 @@ const IAICanvasMaskOptions = () => {
<FormControlGroup formLabelProps={formLabelProps}> <FormControlGroup formLabelProps={formLabelProps}>
<FormControl> <FormControl>
<FormLabel>{`${t('unifiedCanvas.enableMask')} (H)`}</FormLabel> <FormLabel>{`${t('unifiedCanvas.enableMask')} (H)`}</FormLabel>
<Checkbox <Checkbox isChecked={isMaskEnabled} onChange={handleToggleEnableMask} />
isChecked={isMaskEnabled}
onChange={handleToggleEnableMask}
/>
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.preserveMaskedArea')}</FormLabel> <FormLabel>{t('unifiedCanvas.preserveMaskedArea')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldPreserveMaskedArea} onChange={handleChangePreserveMaskedArea} />
isChecked={shouldPreserveMaskedArea}
onChange={handleChangePreserveMaskedArea}
/>
</FormControl> </FormControl>
</FormControlGroup> </FormControlGroup>
<Box pt={2} pb={2}> <Box pt={2} pb={2}>
<IAIColorPicker <IAIColorPicker color={maskColor} onChange={handleChangeMaskColor} />
color={maskColor}
onChange={handleChangeMaskColor}
/>
</Box> </Box>
<ButtonGroup isAttached={false}> <ButtonGroup isAttached={false}>
<Button <Button size="sm" leftIcon={<PiFloppyDiskBackFill />} onClick={handleSaveMask}>
size="sm"
leftIcon={<PiFloppyDiskBackFill />}
onClick={handleSaveMask}
>
{t('unifiedCanvas.saveMask')} {t('unifiedCanvas.saveMask')}
</Button> </Button>
<Button <Button size="sm" leftIcon={<PiTrashSimpleFill />} onClick={handleClearMask}>
size="sm"
leftIcon={<PiTrashSimpleFill />}
onClick={handleClearMask}
>
{t('unifiedCanvas.clearMask')} {t('unifiedCanvas.clearMask')}
</Button> </Button>
</ButtonGroup> </ButtonGroup>

View File

@ -38,23 +38,13 @@ const IAICanvasSettingsButtonPopover = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const { t } = useTranslation(); const { t } = useTranslation();
const shouldAutoSave = useAppSelector((s) => s.canvas.shouldAutoSave); const shouldAutoSave = useAppSelector((s) => s.canvas.shouldAutoSave);
const shouldCropToBoundingBoxOnSave = useAppSelector( const shouldCropToBoundingBoxOnSave = useAppSelector((s) => s.canvas.shouldCropToBoundingBoxOnSave);
(s) => s.canvas.shouldCropToBoundingBoxOnSave const shouldDarkenOutsideBoundingBox = useAppSelector((s) => s.canvas.shouldDarkenOutsideBoundingBox);
); const shouldShowCanvasDebugInfo = useAppSelector((s) => s.canvas.shouldShowCanvasDebugInfo);
const shouldDarkenOutsideBoundingBox = useAppSelector(
(s) => s.canvas.shouldDarkenOutsideBoundingBox
);
const shouldShowCanvasDebugInfo = useAppSelector(
(s) => s.canvas.shouldShowCanvasDebugInfo
);
const shouldShowGrid = useAppSelector((s) => s.canvas.shouldShowGrid); const shouldShowGrid = useAppSelector((s) => s.canvas.shouldShowGrid);
const shouldShowIntermediates = useAppSelector( const shouldShowIntermediates = useAppSelector((s) => s.canvas.shouldShowIntermediates);
(s) => s.canvas.shouldShowIntermediates
);
const shouldSnapToGrid = useAppSelector((s) => s.canvas.shouldSnapToGrid); const shouldSnapToGrid = useAppSelector((s) => s.canvas.shouldSnapToGrid);
const shouldRestrictStrokesToBox = useAppSelector( const shouldRestrictStrokesToBox = useAppSelector((s) => s.canvas.shouldRestrictStrokesToBox);
(s) => s.canvas.shouldRestrictStrokesToBox
);
const shouldAntialias = useAppSelector((s) => s.canvas.shouldAntialias); const shouldAntialias = useAppSelector((s) => s.canvas.shouldAntialias);
useHotkeys( useHotkeys(
@ -70,49 +60,40 @@ const IAICanvasSettingsButtonPopover = () => {
); );
const handleChangeShouldSnapToGrid = useCallback( const handleChangeShouldSnapToGrid = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldSnapToGrid(e.target.checked)),
dispatch(setShouldSnapToGrid(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldShowIntermediates = useCallback( const handleChangeShouldShowIntermediates = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldShowIntermediates(e.target.checked)),
dispatch(setShouldShowIntermediates(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldShowGrid = useCallback( const handleChangeShouldShowGrid = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldShowGrid(e.target.checked)),
dispatch(setShouldShowGrid(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldDarkenOutsideBoundingBox = useCallback( const handleChangeShouldDarkenOutsideBoundingBox = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldDarkenOutsideBoundingBox(e.target.checked)),
dispatch(setShouldDarkenOutsideBoundingBox(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldAutoSave = useCallback( const handleChangeShouldAutoSave = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldAutoSave(e.target.checked)),
dispatch(setShouldAutoSave(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldCropToBoundingBoxOnSave = useCallback( const handleChangeShouldCropToBoundingBoxOnSave = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldCropToBoundingBoxOnSave(e.target.checked)),
dispatch(setShouldCropToBoundingBoxOnSave(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldRestrictStrokesToBox = useCallback( const handleChangeShouldRestrictStrokesToBox = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldRestrictStrokesToBox(e.target.checked)),
dispatch(setShouldRestrictStrokesToBox(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldShowCanvasDebugInfo = useCallback( const handleChangeShouldShowCanvasDebugInfo = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldShowCanvasDebugInfo(e.target.checked)),
dispatch(setShouldShowCanvasDebugInfo(e.target.checked)),
[dispatch] [dispatch]
); );
const handleChangeShouldAntialias = useCallback( const handleChangeShouldAntialias = useCallback(
(e: ChangeEvent<HTMLInputElement>) => (e: ChangeEvent<HTMLInputElement>) => dispatch(setShouldAntialias(e.target.checked)),
dispatch(setShouldAntialias(e.target.checked)),
[dispatch] [dispatch]
); );
@ -131,29 +112,18 @@ const IAICanvasSettingsButtonPopover = () => {
<FormControlGroup formLabelProps={formLabelProps}> <FormControlGroup formLabelProps={formLabelProps}>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.showIntermediates')}</FormLabel> <FormLabel>{t('unifiedCanvas.showIntermediates')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldShowIntermediates} onChange={handleChangeShouldShowIntermediates} />
isChecked={shouldShowIntermediates}
onChange={handleChangeShouldShowIntermediates}
/>
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.showGrid')}</FormLabel> <FormLabel>{t('unifiedCanvas.showGrid')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldShowGrid} onChange={handleChangeShouldShowGrid} />
isChecked={shouldShowGrid}
onChange={handleChangeShouldShowGrid}
/>
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.snapToGrid')}</FormLabel> <FormLabel>{t('unifiedCanvas.snapToGrid')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldSnapToGrid} onChange={handleChangeShouldSnapToGrid} />
isChecked={shouldSnapToGrid}
onChange={handleChangeShouldSnapToGrid}
/>
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel> <FormLabel>{t('unifiedCanvas.darkenOutsideSelection')}</FormLabel>
{t('unifiedCanvas.darkenOutsideSelection')}
</FormLabel>
<Checkbox <Checkbox
isChecked={shouldDarkenOutsideBoundingBox} isChecked={shouldDarkenOutsideBoundingBox}
onChange={handleChangeShouldDarkenOutsideBoundingBox} onChange={handleChangeShouldDarkenOutsideBoundingBox}
@ -161,10 +131,7 @@ const IAICanvasSettingsButtonPopover = () => {
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.autoSaveToGallery')}</FormLabel> <FormLabel>{t('unifiedCanvas.autoSaveToGallery')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldAutoSave} onChange={handleChangeShouldAutoSave} />
isChecked={shouldAutoSave}
onChange={handleChangeShouldAutoSave}
/>
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.saveBoxRegionOnly')}</FormLabel> <FormLabel>{t('unifiedCanvas.saveBoxRegionOnly')}</FormLabel>
@ -175,24 +142,15 @@ const IAICanvasSettingsButtonPopover = () => {
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.limitStrokesToBox')}</FormLabel> <FormLabel>{t('unifiedCanvas.limitStrokesToBox')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldRestrictStrokesToBox} onChange={handleChangeShouldRestrictStrokesToBox} />
isChecked={shouldRestrictStrokesToBox}
onChange={handleChangeShouldRestrictStrokesToBox}
/>
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.showCanvasDebugInfo')}</FormLabel> <FormLabel>{t('unifiedCanvas.showCanvasDebugInfo')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldShowCanvasDebugInfo} onChange={handleChangeShouldShowCanvasDebugInfo} />
isChecked={shouldShowCanvasDebugInfo}
onChange={handleChangeShouldShowCanvasDebugInfo}
/>
</FormControl> </FormControl>
<FormControl> <FormControl>
<FormLabel>{t('unifiedCanvas.antialiasing')}</FormLabel> <FormLabel>{t('unifiedCanvas.antialiasing')}</FormLabel>
<Checkbox <Checkbox isChecked={shouldAntialias} onChange={handleChangeShouldAntialias} />
isChecked={shouldAntialias}
onChange={handleChangeShouldAntialias}
/>
</FormControl> </FormControl>
</FormControlGroup> </FormControlGroup>
<ClearCanvasHistoryButtonModal /> <ClearCanvasHistoryButtonModal />

View File

@ -15,17 +15,9 @@ import {
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import IAIColorPicker from 'common/components/IAIColorPicker'; import IAIColorPicker from 'common/components/IAIColorPicker';
import { import { $tool, resetToolInteractionState } from 'features/canvas/store/canvasNanostore';
$tool,
resetToolInteractionState,
} from 'features/canvas/store/canvasNanostore';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { import { addEraseRect, addFillRect, setBrushColor, setBrushSize } from 'features/canvas/store/canvasSlice';
addEraseRect,
addFillRect,
setBrushColor,
setBrushSize,
} from 'features/canvas/store/canvasSlice';
import { clamp } from 'lodash-es'; import { clamp } from 'lodash-es';
import { memo, useCallback } from 'react'; import { memo, useCallback } from 'react';
import type { RgbaColor } from 'react-colorful'; import type { RgbaColor } from 'react-colorful';
@ -275,11 +267,7 @@ const IAICanvasToolChooserOptions = () => {
</FormControl> </FormControl>
</Flex> </Flex>
<Box w="full" pt={2} pb={2}> <Box w="full" pt={2} pb={2}>
<IAIColorPicker <IAIColorPicker color={brushColor} onChange={handleChangeBrushColor} withNumberInput />
color={brushColor}
onChange={handleChangeBrushColor}
withNumberInput
/>
</Box> </Box>
</Flex> </Flex>
</PopoverBody> </PopoverBody>

View File

@ -1,12 +1,5 @@
import type { ComboboxOnChange } from '@invoke-ai/ui-library'; import type { ComboboxOnChange } from '@invoke-ai/ui-library';
import { import { ButtonGroup, Combobox, Flex, FormControl, IconButton, Tooltip } from '@invoke-ai/ui-library';
ButtonGroup,
Combobox,
Flex,
FormControl,
IconButton,
Tooltip,
} from '@invoke-ai/ui-library';
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { useCopyImageToClipboard } from 'common/hooks/useCopyImageToClipboard'; import { useCopyImageToClipboard } from 'common/hooks/useCopyImageToClipboard';
@ -20,12 +13,7 @@ import {
} from 'features/canvas/store/actions'; } from 'features/canvas/store/actions';
import { $canvasBaseLayer, $tool } from 'features/canvas/store/canvasNanostore'; import { $canvasBaseLayer, $tool } from 'features/canvas/store/canvasNanostore';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { import { resetCanvas, resetCanvasView, setIsMaskEnabled, setLayer } from 'features/canvas/store/canvasSlice';
resetCanvas,
resetCanvasView,
setIsMaskEnabled,
setLayer,
} from 'features/canvas/store/canvasSlice';
import type { CanvasLayer } from 'features/canvas/store/canvasTypes'; import type { CanvasLayer } from 'features/canvas/store/canvasTypes';
import { LAYER_NAMES_DICT } from 'features/canvas/store/canvasTypes'; import { LAYER_NAMES_DICT } from 'features/canvas/store/canvasTypes';
import { memo, useCallback, useMemo } from 'react'; import { memo, useCallback, useMemo } from 'react';
@ -203,20 +191,13 @@ const IAICanvasToolbar = () => {
[dispatch, isMaskEnabled] [dispatch, isMaskEnabled]
); );
const value = useMemo( const value = useMemo(() => LAYER_NAMES_DICT.filter((o) => o.value === layer)[0], [layer]);
() => LAYER_NAMES_DICT.filter((o) => o.value === layer)[0],
[layer]
);
return ( return (
<Flex alignItems="center" gap={2} flexWrap="wrap"> <Flex alignItems="center" gap={2} flexWrap="wrap">
<Tooltip label={`${t('unifiedCanvas.layer')} (Q)`}> <Tooltip label={`${t('unifiedCanvas.layer')} (Q)`}>
<FormControl isDisabled={isStaging} w="5rem"> <FormControl isDisabled={isStaging} w="5rem">
<Combobox <Combobox value={value} options={LAYER_NAMES_DICT} onChange={handleChangeLayer} />
value={value}
options={LAYER_NAMES_DICT}
onChange={handleChangeLayer}
/>
</FormControl> </FormControl>
</Tooltip> </Tooltip>

View File

@ -1,9 +1,5 @@
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { import { $isMovingBoundingBox, $isMovingStage, $tool } from 'features/canvas/store/canvasNanostore';
$isMovingBoundingBox,
$isMovingStage,
$tool,
} from 'features/canvas/store/canvasNanostore';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { setStageCoordinates } from 'features/canvas/store/canvasSlice'; import { setStageCoordinates } from 'features/canvas/store/canvasSlice';
import type { KonvaEventObject } from 'konva/lib/Node'; import type { KonvaEventObject } from 'konva/lib/Node';
@ -13,9 +9,7 @@ const useCanvasDrag = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const isStaging = useAppSelector(isStagingSelector); const isStaging = useAppSelector(isStagingSelector);
const handleDragStart = useCallback(() => { const handleDragStart = useCallback(() => {
if ( if (!(($tool.get() === 'move' || isStaging) && !$isMovingBoundingBox.get())) {
!(($tool.get() === 'move' || isStaging) && !$isMovingBoundingBox.get())
) {
return; return;
} }
$isMovingStage.set(true); $isMovingStage.set(true);
@ -36,9 +30,7 @@ const useCanvasDrag = () => {
); );
const handleDragEnd = useCallback(() => { const handleDragEnd = useCallback(() => {
if ( if (!(($tool.get() === 'move' || isStaging) && !$isMovingBoundingBox.get())) {
!(($tool.get() === 'move' || isStaging) && !$isMovingBoundingBox.get())
) {
return; return;
} }
$isMovingStage.set(false); $isMovingStage.set(false);

View File

@ -8,30 +8,16 @@ import { useDebounce } from 'react-use';
export const useCanvasGenerationMode = () => { export const useCanvasGenerationMode = () => {
const layerState = useAppSelector((s) => s.canvas.layerState); const layerState = useAppSelector((s) => s.canvas.layerState);
const boundingBoxCoordinates = useAppSelector( const boundingBoxCoordinates = useAppSelector((s) => s.canvas.boundingBoxCoordinates);
(s) => s.canvas.boundingBoxCoordinates const boundingBoxDimensions = useAppSelector((s) => s.canvas.boundingBoxDimensions);
);
const boundingBoxDimensions = useAppSelector(
(s) => s.canvas.boundingBoxDimensions
);
const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled); const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled);
const shouldPreserveMaskedArea = useAppSelector( const shouldPreserveMaskedArea = useAppSelector((s) => s.canvas.shouldPreserveMaskedArea);
(s) => s.canvas.shouldPreserveMaskedArea const [generationMode, setGenerationMode] = useState<GenerationMode | undefined>();
);
const [generationMode, setGenerationMode] = useState<
GenerationMode | undefined
>();
useEffect(() => { useEffect(() => {
setGenerationMode(undefined); setGenerationMode(undefined);
}, [ }, [layerState, boundingBoxCoordinates, boundingBoxDimensions, isMaskEnabled, shouldPreserveMaskedArea]);
layerState,
boundingBoxCoordinates,
boundingBoxDimensions,
isMaskEnabled,
shouldPreserveMaskedArea,
]);
useDebounce( useDebounce(
async () => { async () => {
@ -51,21 +37,12 @@ export const useCanvasGenerationMode = () => {
const { baseImageData, maskImageData } = canvasBlobsAndImageData; const { baseImageData, maskImageData } = canvasBlobsAndImageData;
// Determine the generation mode // Determine the generation mode
const generationMode = getCanvasGenerationMode( const generationMode = getCanvasGenerationMode(baseImageData, maskImageData);
baseImageData,
maskImageData
);
setGenerationMode(generationMode); setGenerationMode(generationMode);
}, },
1000, 1000,
[ [layerState, boundingBoxCoordinates, boundingBoxDimensions, isMaskEnabled, shouldPreserveMaskedArea]
layerState,
boundingBoxCoordinates,
boundingBoxDimensions,
isMaskEnabled,
shouldPreserveMaskedArea,
]
); );
return generationMode; return generationMode;

View File

@ -22,9 +22,7 @@ import { useHotkeys } from 'react-hotkeys-hook';
const useInpaintingCanvasHotkeys = () => { const useInpaintingCanvasHotkeys = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const activeTabName = useAppSelector(activeTabNameSelector); const activeTabName = useAppSelector(activeTabNameSelector);
const shouldShowBoundingBox = useAppSelector( const shouldShowBoundingBox = useAppSelector((s) => s.canvas.shouldShowBoundingBox);
(s) => s.canvas.shouldShowBoundingBox
);
const isStaging = useAppSelector(isStagingSelector); const isStaging = useAppSelector(isStagingSelector);
const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled); const isMaskEnabled = useAppSelector((s) => s.canvas.isMaskEnabled);
const shouldSnapToGrid = useAppSelector((s) => s.canvas.shouldSnapToGrid); const shouldSnapToGrid = useAppSelector((s) => s.canvas.shouldSnapToGrid);
@ -44,8 +42,7 @@ const useInpaintingCanvasHotkeys = () => {
[] []
); );
const handleToggleEnableMask = () => const handleToggleEnableMask = () => dispatch(setIsMaskEnabled(!isMaskEnabled));
dispatch(setIsMaskEnabled(!isMaskEnabled));
useHotkeys( useHotkeys(
['h'], ['h'],

View File

@ -1,9 +1,5 @@
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { import { $isDrawing, $isMovingStage, $tool } from 'features/canvas/store/canvasNanostore';
$isDrawing,
$isMovingStage,
$tool,
} from 'features/canvas/store/canvasNanostore';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { addLine } from 'features/canvas/store/canvasSlice'; import { addLine } from 'features/canvas/store/canvasSlice';
import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition'; import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition';

View File

@ -1,9 +1,5 @@
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { import { $cursorPosition, $isDrawing, $tool } from 'features/canvas/store/canvasNanostore';
$cursorPosition,
$isDrawing,
$tool,
} from 'features/canvas/store/canvasNanostore';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { addPointToCurrentLine } from 'features/canvas/store/canvasSlice'; import { addPointToCurrentLine } from 'features/canvas/store/canvasSlice';
import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition'; import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition';
@ -49,17 +45,8 @@ const useCanvasMouseMove = (
} }
didMouseMoveRef.current = true; didMouseMoveRef.current = true;
dispatch( dispatch(addPointToCurrentLine([scaledCursorPosition.x, scaledCursorPosition.y]));
addPointToCurrentLine([scaledCursorPosition.x, scaledCursorPosition.y]) }, [didMouseMoveRef, dispatch, isStaging, lastCursorPositionRef, stageRef, updateColorUnderCursor]);
);
}, [
didMouseMoveRef,
dispatch,
isStaging,
lastCursorPositionRef,
stageRef,
updateColorUnderCursor,
]);
}; };
export default useCanvasMouseMove; export default useCanvasMouseMove;

View File

@ -1,10 +1,6 @@
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { import { $isDrawing, $isMovingStage, $tool } from 'features/canvas/store/canvasNanostore';
$isDrawing,
$isMovingStage,
$tool,
} from 'features/canvas/store/canvasNanostore';
import { isStagingSelector } from 'features/canvas/store/canvasSelectors'; import { isStagingSelector } from 'features/canvas/store/canvasSelectors';
import { addPointToCurrentLine } from 'features/canvas/store/canvasSlice'; import { addPointToCurrentLine } from 'features/canvas/store/canvasSlice';
import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition'; import getScaledCursorPosition from 'features/canvas/util/getScaledCursorPosition';
@ -39,9 +35,7 @@ const useCanvasMouseUp = (
* the line's existing points. This allows the line to render as a circle * the line's existing points. This allows the line to render as a circle
* centered on that point. * centered on that point.
*/ */
dispatch( dispatch(addPointToCurrentLine([scaledCursorPosition.x, scaledCursorPosition.y]));
addPointToCurrentLine([scaledCursorPosition.x, scaledCursorPosition.y])
);
} else { } else {
didMouseMoveRef.current = false; didMouseMoveRef.current = false;
} }

View File

@ -1,15 +1,8 @@
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks'; import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
import { $isMoveStageKeyHeld } from 'features/canvas/store/canvasNanostore'; import { $isMoveStageKeyHeld } from 'features/canvas/store/canvasNanostore';
import { import { setStageCoordinates, setStageScale } from 'features/canvas/store/canvasSlice';
setStageCoordinates, import { CANVAS_SCALE_BY, MAX_CANVAS_SCALE, MIN_CANVAS_SCALE } from 'features/canvas/util/constants';
setStageScale,
} from 'features/canvas/store/canvasSlice';
import {
CANVAS_SCALE_BY,
MAX_CANVAS_SCALE,
MIN_CANVAS_SCALE,
} from 'features/canvas/util/constants';
import type Konva from 'konva'; import type Konva from 'konva';
import type { KonvaEventObject } from 'konva/lib/Node'; import type { KonvaEventObject } from 'konva/lib/Node';
import { clamp } from 'lodash-es'; import { clamp } from 'lodash-es';
@ -49,11 +42,7 @@ const useCanvasWheel = (stageRef: MutableRefObject<Konva.Stage | null>) => {
delta = -delta; delta = -delta;
} }
const newScale = clamp( const newScale = clamp(stageScale * CANVAS_SCALE_BY ** delta, MIN_CANVAS_SCALE, MAX_CANVAS_SCALE);
stageScale * CANVAS_SCALE_BY ** delta,
MIN_CANVAS_SCALE,
MAX_CANVAS_SCALE
);
const newCoordinates = { const newCoordinates = {
x: cursorPos.x - mousePointTo.x * newScale, x: cursorPos.x - mousePointTo.x * newScale,

View File

@ -1,13 +1,6 @@
import { useAppDispatch } from 'app/store/storeHooks'; import { useAppDispatch } from 'app/store/storeHooks';
import { import { $canvasBaseLayer, $canvasStage, $tool } from 'features/canvas/store/canvasNanostore';
$canvasBaseLayer, import { commitColorPickerColor, setColorPickerColor } from 'features/canvas/store/canvasSlice';
$canvasStage,
$tool,
} from 'features/canvas/store/canvasNanostore';
import {
commitColorPickerColor,
setColorPickerColor,
} from 'features/canvas/store/canvasSlice';
import Konva from 'konva'; import Konva from 'konva';
import { useCallback } from 'react'; import { useCallback } from 'react';
@ -31,19 +24,9 @@ const useColorPicker = () => {
const [r, g, b, a] = canvasBaseLayer const [r, g, b, a] = canvasBaseLayer
.getContext() .getContext()
.getImageData( .getImageData(position.x * pixelRatio, position.y * pixelRatio, 1, 1).data;
position.x * pixelRatio,
position.y * pixelRatio,
1,
1
).data;
if ( if (r === undefined || g === undefined || b === undefined || a === undefined) {
r === undefined ||
g === undefined ||
b === undefined ||
a === undefined
) {
return; return;
} }

Some files were not shown because too many files have changed in this diff Show More