diff --git a/invokeai/frontend/web/src/app/store/middleware/devtools/actionsDenylist.ts b/invokeai/frontend/web/src/app/store/middleware/devtools/actionsDenylist.ts index eb54868735..8a6e112d27 100644 --- a/invokeai/frontend/web/src/app/store/middleware/devtools/actionsDenylist.ts +++ b/invokeai/frontend/web/src/app/store/middleware/devtools/actionsDenylist.ts @@ -9,4 +9,5 @@ export const actionsDenylist = [ 'canvas/addPointToCurrentLine', 'socket/socketGeneratorProgress', 'socket/appSocketGeneratorProgress', + 'hotkeys/shiftKeyPressed', ]; diff --git a/invokeai/frontend/web/src/common/components/IAIInput.tsx b/invokeai/frontend/web/src/common/components/IAIInput.tsx index 3cba36d2c9..d114fc5968 100644 --- a/invokeai/frontend/web/src/common/components/IAIInput.tsx +++ b/invokeai/frontend/web/src/common/components/IAIInput.tsx @@ -5,8 +5,10 @@ import { Input, InputProps, } from '@chakra-ui/react'; +import { useAppDispatch } from 'app/store/storeHooks'; import { stopPastePropagation } from 'common/util/stopPastePropagation'; -import { ChangeEvent, memo } from 'react'; +import { shiftKeyPressed } from 'features/ui/store/hotkeysSlice'; +import { ChangeEvent, KeyboardEvent, memo, useCallback } from 'react'; interface IAIInputProps extends InputProps { label?: string; @@ -25,6 +27,25 @@ const IAIInput = (props: IAIInputProps) => { ...rest } = props; + const dispatch = useAppDispatch(); + const handleKeyDown = useCallback( + (e: KeyboardEvent) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + return ( { {...formControlProps} > {label !== '' && {label}} - + ); }; diff --git a/invokeai/frontend/web/src/common/components/IAIMantineMultiSelect.tsx b/invokeai/frontend/web/src/common/components/IAIMantineMultiSelect.tsx index 04bab3717a..e52ec63810 100644 --- a/invokeai/frontend/web/src/common/components/IAIMantineMultiSelect.tsx +++ b/invokeai/frontend/web/src/common/components/IAIMantineMultiSelect.tsx @@ -1,7 +1,9 @@ import { Tooltip, useColorMode, useToken } from '@chakra-ui/react'; import { MultiSelect, MultiSelectProps } from '@mantine/core'; +import { useAppDispatch } from 'app/store/storeHooks'; import { useChakraThemeTokens } from 'common/hooks/useChakraThemeTokens'; -import { RefObject, memo } from 'react'; +import { shiftKeyPressed } from 'features/ui/store/hotkeysSlice'; +import { KeyboardEvent, RefObject, memo, useCallback } from 'react'; import { mode } from 'theme/util/mode'; type IAIMultiSelectProps = MultiSelectProps & { @@ -11,6 +13,7 @@ type IAIMultiSelectProps = MultiSelectProps & { const IAIMantineMultiSelect = (props: IAIMultiSelectProps) => { const { searchable = true, tooltip, inputRef, ...rest } = props; + const dispatch = useAppDispatch(); const { base50, base100, @@ -31,10 +34,30 @@ const IAIMantineMultiSelect = (props: IAIMultiSelectProps) => { const [boxShadow] = useToken('shadows', ['dark-lg']); const { colorMode } = useColorMode(); + const handleKeyDown = useCallback( + (e: KeyboardEvent) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + return ( ({ label: { diff --git a/invokeai/frontend/web/src/common/components/IAIMantineSelect.tsx b/invokeai/frontend/web/src/common/components/IAIMantineSelect.tsx index 8469af8fc8..80e6c24ace 100644 --- a/invokeai/frontend/web/src/common/components/IAIMantineSelect.tsx +++ b/invokeai/frontend/web/src/common/components/IAIMantineSelect.tsx @@ -1,7 +1,9 @@ import { Tooltip, useColorMode, useToken } from '@chakra-ui/react'; import { Select, SelectProps } from '@mantine/core'; +import { useAppDispatch } from 'app/store/storeHooks'; import { useChakraThemeTokens } from 'common/hooks/useChakraThemeTokens'; -import { memo } from 'react'; +import { shiftKeyPressed } from 'features/ui/store/hotkeysSlice'; +import { KeyboardEvent, memo, useCallback } from 'react'; import { mode } from 'theme/util/mode'; export type IAISelectDataType = { @@ -16,6 +18,7 @@ type IAISelectProps = SelectProps & { const IAIMantineSelect = (props: IAISelectProps) => { const { searchable = true, tooltip, ...rest } = props; + const dispatch = useAppDispatch(); const { base50, base100, @@ -36,11 +39,31 @@ const IAIMantineSelect = (props: IAISelectProps) => { const { colorMode } = useColorMode(); + const handleKeyDown = useCallback( + (e: KeyboardEvent) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + const [boxShadow] = useToken('shadows', ['dark-lg']); return (