From a0ccb4385fe8c9999bf815b6e9ca4570721e9d1c Mon Sep 17 00:00:00 2001 From: psychedelicious <4822129+psychedelicious@users.noreply.github.com> Date: Sat, 8 Jul 2023 18:52:37 +1000 Subject: [PATCH] fix(ui): fix inconsistent shift modifier capture The shift key listener didn't catch pressed when focused in a textarea or input field, causing jank on slider number inputs. Add keydown and keyup listeners to all such fields, which ensures that the `shift` state is always correct. Also add the action tracking it to `actionsDenylist` to not clutter up devtools. --- .../middleware/devtools/actionsDenylist.ts | 1 + .../web/src/common/components/IAIInput.tsx | 30 ++++++++++++++- .../components/IAIMantineMultiSelect.tsx | 25 ++++++++++++- .../common/components/IAIMantineSelect.tsx | 25 ++++++++++++- .../src/common/components/IAINumberInput.tsx | 37 ++++++++++++++++++- .../web/src/common/components/IAISlider.tsx | 25 ++++++++++++- .../web/src/common/components/IAITextarea.tsx | 33 ++++++++++++++++- 7 files changed, 167 insertions(+), 9 deletions(-) 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<HTMLInputElement>) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent<HTMLInputElement>) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + return ( <FormControl isInvalid={isInvalid} @@ -32,7 +53,12 @@ const IAIInput = (props: IAIInputProps) => { {...formControlProps} > {label !== '' && <FormLabel>{label}</FormLabel>} - <Input {...rest} onPaste={stopPastePropagation} /> + <Input + {...rest} + onPaste={stopPastePropagation} + onKeyDown={handleKeyDown} + onKeyUp={handleKeyUp} + /> </FormControl> ); }; 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<HTMLInputElement>) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent<HTMLInputElement>) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + return ( <Tooltip label={tooltip} placement="top" hasArrow isOpen={true}> <MultiSelect ref={inputRef} + onKeyDown={handleKeyDown} + onKeyUp={handleKeyUp} searchable={searchable} styles={() => ({ 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<HTMLInputElement>) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent<HTMLInputElement>) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + const [boxShadow] = useToken('shadows', ['dark-lg']); return ( <Tooltip label={tooltip} placement="top" hasArrow> <Select + onKeyDown={handleKeyDown} + onKeyUp={handleKeyUp} searchable={searchable} styles={() => ({ label: { diff --git a/invokeai/frontend/web/src/common/components/IAINumberInput.tsx b/invokeai/frontend/web/src/common/components/IAINumberInput.tsx index bf598f3b12..8f675cc148 100644 --- a/invokeai/frontend/web/src/common/components/IAINumberInput.tsx +++ b/invokeai/frontend/web/src/common/components/IAINumberInput.tsx @@ -14,10 +14,19 @@ import { Tooltip, TooltipProps, } from '@chakra-ui/react'; +import { useAppDispatch } from 'app/store/storeHooks'; import { stopPastePropagation } from 'common/util/stopPastePropagation'; +import { shiftKeyPressed } from 'features/ui/store/hotkeysSlice'; import { clamp } from 'lodash-es'; -import { FocusEvent, memo, useEffect, useState } from 'react'; +import { + FocusEvent, + KeyboardEvent, + memo, + useCallback, + useEffect, + useState, +} from 'react'; const numberStringRegex = /^-?(0\.)?\.?$/; @@ -60,6 +69,8 @@ const IAINumberInput = (props: Props) => { ...rest } = props; + const dispatch = useAppDispatch(); + /** * Using a controlled input with a value that accepts decimals needs special * handling. If the user starts to type in "1.5", by the time they press the @@ -109,6 +120,24 @@ const IAINumberInput = (props: Props) => { onChange(clamped); }; + const handleKeyDown = useCallback( + (e: KeyboardEvent<HTMLInputElement>) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent<HTMLInputElement>) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + return ( <Tooltip {...tooltipProps}> <FormControl @@ -128,7 +157,11 @@ const IAINumberInput = (props: Props) => { {...rest} onPaste={stopPastePropagation} > - <NumberInputField {...numberInputFieldProps} /> + <NumberInputField + {...numberInputFieldProps} + onKeyDown={handleKeyDown} + onKeyUp={handleKeyUp} + /> {showStepper && ( <NumberInputStepper> <NumberIncrementStepper {...numberInputStepperProps} /> diff --git a/invokeai/frontend/web/src/common/components/IAISlider.tsx b/invokeai/frontend/web/src/common/components/IAISlider.tsx index 49ea980612..d99fbfa149 100644 --- a/invokeai/frontend/web/src/common/components/IAISlider.tsx +++ b/invokeai/frontend/web/src/common/components/IAISlider.tsx @@ -26,9 +26,12 @@ import { } from '@chakra-ui/react'; import { clamp } from 'lodash-es'; +import { useAppDispatch } from 'app/store/storeHooks'; import { roundDownToMultiple } from 'common/util/roundDownToMultiple'; +import { shiftKeyPressed } from 'features/ui/store/hotkeysSlice'; import { FocusEvent, + KeyboardEvent, memo, MouseEvent, useCallback, @@ -107,7 +110,7 @@ const IAISlider = (props: IAIFullSliderProps) => { sliderIAIIconButtonProps, ...rest } = props; - + const dispatch = useAppDispatch(); const { t } = useTranslation(); const [localInputValue, setLocalInputValue] = useState< @@ -167,6 +170,24 @@ const IAISlider = (props: IAIFullSliderProps) => { } }, []); + const handleKeyDown = useCallback( + (e: KeyboardEvent<HTMLInputElement>) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent<HTMLInputElement>) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + return ( <FormControl onClick={forceInputBlur} @@ -310,6 +331,8 @@ const IAISlider = (props: IAIFullSliderProps) => { {...sliderNumberInputProps} > <NumberInputField + onKeyDown={handleKeyDown} + onKeyUp={handleKeyUp} minWidth={inputWidth} {...sliderNumberInputFieldProps} /> diff --git a/invokeai/frontend/web/src/common/components/IAITextarea.tsx b/invokeai/frontend/web/src/common/components/IAITextarea.tsx index b5247887bb..e29c6fe513 100644 --- a/invokeai/frontend/web/src/common/components/IAITextarea.tsx +++ b/invokeai/frontend/web/src/common/components/IAITextarea.tsx @@ -1,9 +1,38 @@ import { Textarea, TextareaProps, forwardRef } from '@chakra-ui/react'; +import { useAppDispatch } from 'app/store/storeHooks'; import { stopPastePropagation } from 'common/util/stopPastePropagation'; -import { memo } from 'react'; +import { shiftKeyPressed } from 'features/ui/store/hotkeysSlice'; +import { KeyboardEvent, memo, useCallback } from 'react'; const IAITextarea = forwardRef((props: TextareaProps, ref) => { - return <Textarea ref={ref} onPaste={stopPastePropagation} {...props} />; + const dispatch = useAppDispatch(); + const handleKeyDown = useCallback( + (e: KeyboardEvent<HTMLTextAreaElement>) => { + if (e.shiftKey) { + dispatch(shiftKeyPressed(true)); + } + }, + [dispatch] + ); + + const handleKeyUp = useCallback( + (e: KeyboardEvent<HTMLTextAreaElement>) => { + if (!e.shiftKey) { + dispatch(shiftKeyPressed(false)); + } + }, + [dispatch] + ); + + return ( + <Textarea + ref={ref} + onPaste={stopPastePropagation} + onKeyDown={handleKeyDown} + onKeyUp={handleKeyUp} + {...props} + /> + ); }); export default memo(IAITextarea);