From 85c1430a96cd534cbfc6c3764c1488c2cdb83601 Mon Sep 17 00:00:00 2001 From: Iajret Creature <122297233+Steals-The-PRs@users.noreply.github.com> Date: Sun, 24 Mar 2024 20:16:56 +0300 Subject: [PATCH] [MIRROR] Typescript refactor for Number Input (#1429) (#2545) * Typescript refactor for Number Input (#81913) ## About The Pull Request - Fixes #80971 - Fixes #81379 You now only click once to highlight & edit the value. Converted NumberInput to typescript ## Changelog :cl: fix: NumberInput(used in chem heater, plumbing reaction chamber etc) highlighting & editing requires only single click like before refactor: Typescript conversion for NumberInput TGUI Component /:cl: --------- * Typescript refactor for Number Input --------- Co-authored-by: NovaBot <154629622+NovaBot13@users.noreply.github.com> Co-authored-by: SyncIt21 <110812394+SyncIt21@users.noreply.github.com> Co-authored-by: Jeremiah <42397676+jlsnow301@users.noreply.github.com> --- tgui/packages/tgui/components/NumberInput.jsx | 288 --------------- tgui/packages/tgui/components/NumberInput.tsx | 337 ++++++++++++++++++ tgui/packages/tgui/interfaces/BountyBoard.tsx | 2 +- .../CentcomPodLauncher/PresetsPage.tsx | 1 - tgui/packages/tgui/interfaces/ChemPress.tsx | 6 +- .../tgui/interfaces/ColorMatrixEditor.tsx | 5 +- .../tgui/interfaces/MatrixMathTester.tsx | 14 + .../tgui/interfaces/Mecha/ModulesPane.tsx | 4 +- .../interfaces/ParticleEdit/EntriesBasic.tsx | 19 +- .../ParticleEdit/EntriesGenerators.tsx | 14 +- .../interfaces/ParticleEdit/Generators.tsx | 15 + tgui/packages/tgui/interfaces/Signaler.tsx | 4 +- 12 files changed, 403 insertions(+), 306 deletions(-) delete mode 100644 tgui/packages/tgui/components/NumberInput.jsx create mode 100644 tgui/packages/tgui/components/NumberInput.tsx diff --git a/tgui/packages/tgui/components/NumberInput.jsx b/tgui/packages/tgui/components/NumberInput.jsx deleted file mode 100644 index f5579731c0b..00000000000 --- a/tgui/packages/tgui/components/NumberInput.jsx +++ /dev/null @@ -1,288 +0,0 @@ -/** - * @file - * @copyright 2020 Aleksej Komarov - * @license MIT - */ - -import { clamp } from 'common/math'; -import { classes } from 'common/react'; -import { Component, createRef } from 'react'; - -import { AnimatedNumber } from './AnimatedNumber'; -import { Box } from './Box'; - -const DEFAULT_UPDATE_RATE = 400; - -export class NumberInput extends Component { - constructor(props) { - super(props); - const { value } = props; - this.inputRef = createRef(); - this.state = { - value, - dragging: false, - editing: false, - internalValue: null, - origin: null, - suppressingFlicker: false, - }; - - // Suppresses flickering while the value propagates through the backend - this.flickerTimer = null; - this.suppressFlicker = () => { - const { suppressFlicker } = this.props; - if (suppressFlicker > 0) { - this.setState({ - suppressingFlicker: true, - }); - clearTimeout(this.flickerTimer); - this.flickerTimer = setTimeout( - () => - this.setState({ - suppressingFlicker: false, - }), - suppressFlicker, - ); - } - }; - - this.handleDragStart = (e) => { - const { value } = this.props; - const { editing } = this.state; - if (editing) { - return; - } - document.body.style['pointer-events'] = 'none'; - this.ref = e.target; - this.setState({ - dragging: false, - origin: e.screenY, - value, - internalValue: value, - }); - this.timer = setTimeout(() => { - this.setState({ - dragging: true, - }); - }, 250); - this.dragInterval = setInterval(() => { - const { dragging, value } = this.state; - const { onDrag } = this.props; - if (dragging && onDrag) { - onDrag(e, value); - } - }, this.props.updateRate || DEFAULT_UPDATE_RATE); - document.addEventListener('mousemove', this.handleDragMove); - document.addEventListener('mouseup', this.handleDragEnd); - }; - - this.handleDragMove = (e) => { - const { minValue, maxValue, step, stepPixelSize } = this.props; - this.setState((prevState) => { - const state = { ...prevState }; - const offset = state.origin - e.screenY; - if (prevState.dragging) { - const stepOffset = Number.isFinite(minValue) ? minValue % step : 0; - // Translate mouse movement to value - // Give it some headroom (by increasing clamp range by 1 step) - state.internalValue = clamp( - state.internalValue + (offset * step) / stepPixelSize, - minValue - step, - maxValue + step, - ); - // Clamp the final value - state.value = clamp( - state.internalValue - (state.internalValue % step) + stepOffset, - minValue, - maxValue, - ); - state.origin = e.screenY; - } else if (Math.abs(offset) > 4) { - state.dragging = true; - } - return state; - }); - }; - - this.handleDragEnd = (e) => { - const { onChange, onDrag } = this.props; - const { dragging, value, internalValue } = this.state; - document.body.style['pointer-events'] = 'auto'; - clearTimeout(this.timer); - clearInterval(this.dragInterval); - this.setState({ - dragging: false, - editing: !dragging, - origin: null, - }); - document.removeEventListener('mousemove', this.handleDragMove); - document.removeEventListener('mouseup', this.handleDragEnd); - if (dragging) { - this.suppressFlicker(); - if (onChange) { - onChange(e, value); - } - if (onDrag) { - onDrag(e, value); - } - } else if (this.inputRef) { - const input = this.inputRef.current; - input.value = internalValue; - // IE8: Dies when trying to focus a hidden element - // (Error: Object does not support this action) - try { - input.focus(); - input.select(); - } catch {} - } - }; - } - - render() { - const { - dragging, - editing, - value: intermediateValue, - suppressingFlicker, - } = this.state; - const { - className, - fluid, - animated, - value, - unit, - minValue, - maxValue, - height, - width, - lineHeight, - fontSize, - format, - onChange, - onDrag, - } = this.props; - let displayValue = value; - if (dragging || suppressingFlicker) { - displayValue = intermediateValue; - } - - const contentElement = ( -
- {animated && !dragging && !suppressingFlicker ? ( - - ) : format ? ( - format(displayValue) - ) : ( - displayValue - )} - - {unit ? ' ' + unit : ''} -
- ); - - return ( - -
-
-
- {contentElement} - { - if (!editing) { - return; - } - const value = clamp(parseFloat(e.target.value), minValue, maxValue); - if (Number.isNaN(value)) { - this.setState({ - editing: false, - }); - return; - } - this.setState({ - editing: false, - value, - }); - this.suppressFlicker(); - if (onChange) { - onChange(e, value); - } - if (onDrag) { - onDrag(e, value); - } - }} - onKeyDown={(e) => { - if (e.keyCode === 13) { - // prettier-ignore - const value = clamp( - parseFloat(e.target.value), - minValue, - maxValue - ); - if (Number.isNaN(value)) { - this.setState({ - editing: false, - }); - return; - } - this.setState({ - editing: false, - value, - }); - this.suppressFlicker(); - if (onChange) { - onChange(e, value); - } - if (onDrag) { - onDrag(e, value); - } - return; - } - if (e.keyCode === 27) { - this.setState({ - editing: false, - }); - return; - } - }} - /> - - ); - } -} - -NumberInput.defaultProps = { - minValue: -Infinity, - maxValue: +Infinity, - step: 1, - stepPixelSize: 1, - suppressFlicker: 50, -}; diff --git a/tgui/packages/tgui/components/NumberInput.tsx b/tgui/packages/tgui/components/NumberInput.tsx new file mode 100644 index 00000000000..dbbcfb62dac --- /dev/null +++ b/tgui/packages/tgui/components/NumberInput.tsx @@ -0,0 +1,337 @@ +import { KEY } from 'common/keys'; +import { clamp } from 'common/math'; +import { BooleanLike, classes } from 'common/react'; +import { + Component, + createRef, + FocusEventHandler, + KeyboardEventHandler, + MouseEventHandler, + RefObject, +} from 'react'; + +import { AnimatedNumber } from './AnimatedNumber'; +import { Box } from './Box'; + +const DEFAULT_UPDATE_RATE = 400; + +type Props = Required<{ + value: number | string; + minValue: number; + maxValue: number; +}> & + Partial<{ + step: number; + stepPixelSize: number; + suppressFlicker: number; + disabled: BooleanLike; + + className: string; + fluid: BooleanLike; + animated: BooleanLike; + unit: string; + height: string; + width: string; + lineHeight: string; + fontSize: string; + updateRate: number; + format: (value: number) => string; + onChange: (e: any, value: number) => void; + onDrag: (e: any, value: number) => void; + }>; + +type State = { + value: number; + dragging: BooleanLike; + editing: BooleanLike; + internalValue: number; + origin: number; + suppressingFlicker: BooleanLike; +}; + +export class NumberInput extends Component { + // Ref to the input field to set focus & highlight + inputRef: RefObject = createRef(); + + // Timer id for the flicker id + flickerTimer: NodeJS.Timeout; + + // After this time has elapsed we are in drag mode so no editing when dragging ends + dragTimeout: NodeJS.Timeout; + + // Call onDrag at this interval + dragInterval: NodeJS.Timeout; + + // default values for the number input state + state: State = { + value: 0, + dragging: false, + editing: false, + internalValue: 0, + origin: 0, + suppressingFlicker: false, + }; + + // default values for the number input props + static defaultProps = { + step: 1, + stepPixelSize: 1, + suppressFlicker: 50, + }; + + constructor(props: Props) { + super(props); + } + + suppressFlicker = () => { + const { suppressFlicker } = this.props; + if (suppressFlicker) { + this.setState({ + suppressingFlicker: true, + }); + clearTimeout(this.flickerTimer); + this.flickerTimer = setTimeout( + () => + this.setState({ + suppressingFlicker: false, + }), + suppressFlicker, + ); + } + }; + + handleDragStart: MouseEventHandler = (event) => { + const { value, updateRate, disabled } = this.props; + const { editing } = this.state; + if (disabled || editing) { + return; + } + this.setState({ + dragging: false, + origin: event.screenY, + internalValue: parseFloat(value.toString()), + }); + this.dragTimeout = setTimeout(() => { + this.setState({ + dragging: true, + }); + }, 250); + + this.dragInterval = setInterval(() => { + const { dragging, value } = this.state; + const { onDrag } = this.props; + if (dragging) { + onDrag?.(event, value); + } + }, updateRate || DEFAULT_UPDATE_RATE); + }; + + handleDragMove: MouseEventHandler = (event) => { + const { minValue, maxValue, step, stepPixelSize, disabled } = this.props; + const { dragging } = this.state; + if (disabled || !dragging) { + return; + } + + this.setState((prevState) => { + const state = { ...prevState }; + const offset = state.origin - event.screenY; + if (prevState.dragging && step) { + const stepOffset = isFinite(minValue) ? minValue % step : 0; + // Translate mouse movement to value + // Give it some headroom (by increasing clamp range by 1 step) + state.internalValue = clamp( + state.internalValue + (offset * step) / (stepPixelSize || 1), + minValue - step, + maxValue + step, + ); + // Clamp the final value + state.value = clamp( + state.internalValue - (state.internalValue % step) + stepOffset, + minValue, + maxValue, + ); + state.origin = event.screenY; + } else if (Math.abs(offset) > 4) { + state.dragging = true; + } + return state; + }); + }; + + handleDragEnd: MouseEventHandler = (event) => { + const { value, dragging, internalValue } = this.state; + const { onDrag, onChange, disabled } = this.props; + if (disabled) { + return; + } + + clearInterval(this.dragInterval); + clearTimeout(this.dragTimeout); + this.setState({ + dragging: false, + editing: !dragging, + }); + + if (dragging) { + this.suppressFlicker(); + onChange?.(event, value); + onDrag?.(event, value); + } else if (this.inputRef) { + const input = this.inputRef.current; + if (input) { + input.value = `${internalValue}`; + setTimeout(() => { + input.focus(); + input.select(); + }, 1); + } + } + }; + + handleBlur: FocusEventHandler = (event) => { + const { editing } = this.state; + const { minValue, maxValue, onChange, onDrag, disabled } = this.props; + + if (disabled || !editing) { + return; + } + + const targetValue = clamp( + parseFloat(event.target.value), + minValue, + maxValue, + ); + if (isNaN(targetValue)) { + this.setState({ + editing: false, + }); + return; + } + this.setState({ + editing: false, + }); + this.suppressFlicker(); + onChange?.(event, targetValue); + onDrag?.(event, targetValue); + }; + + handleKeyDown: KeyboardEventHandler = (event) => { + const { minValue, maxValue, onChange, onDrag, disabled } = this.props; + if (disabled) { + return; + } + + if (event.key === KEY.Enter) { + const targetValue = clamp( + parseFloat(event.currentTarget.value), + minValue, + maxValue, + ); + if (isNaN(targetValue)) { + this.setState({ + editing: false, + }); + return; + } + this.setState({ + editing: false, + value: targetValue, + }); + this.suppressFlicker(); + onChange?.(event, targetValue); + onDrag?.(event, targetValue); + } else if (event.key === KEY.Escape) { + this.setState({ + editing: false, + }); + } + }; + + render() { + const { + dragging, + editing, + value: intermediateValue, + suppressingFlicker, + } = this.state; + + const { + className, + fluid, + animated, + value, + unit, + minValue, + maxValue, + height, + width, + lineHeight, + fontSize, + format, + } = this.props; + + let displayValue = parseFloat(value.toString()); + if (dragging || suppressingFlicker) { + displayValue = intermediateValue; + } + + const contentElement = ( +
+ {animated && !dragging && !suppressingFlicker ? ( + + ) : format ? ( + format(displayValue) + ) : ( + displayValue + )} + + {unit ? ' ' + unit : ''} +
+ ); + + return ( + +
+
+
+ {contentElement} + + + ); + } +} diff --git a/tgui/packages/tgui/interfaces/BountyBoard.tsx b/tgui/packages/tgui/interfaces/BountyBoard.tsx index fcaa3556a70..f0e0eecd86c 100644 --- a/tgui/packages/tgui/interfaces/BountyBoard.tsx +++ b/tgui/packages/tgui/interfaces/BountyBoard.tsx @@ -173,7 +173,7 @@ export const BountyBoardContent = (props) => { /> Hue: setHue(value)} diff --git a/tgui/packages/tgui/interfaces/ChemPress.tsx b/tgui/packages/tgui/interfaces/ChemPress.tsx index f3881061696..6d3c0ead35e 100644 --- a/tgui/packages/tgui/interfaces/ChemPress.tsx +++ b/tgui/packages/tgui/interfaces/ChemPress.tsx @@ -23,10 +23,10 @@ type Category = { }; type Data = { - current_volume: Number; + current_volume: number; product_name: string; - min_volume: Number; - max_volume: Number; + min_volume: number; + max_volume: number; packaging_category: string; packaging_types: Category[]; packaging_type: string; diff --git a/tgui/packages/tgui/interfaces/ColorMatrixEditor.tsx b/tgui/packages/tgui/interfaces/ColorMatrixEditor.tsx index 2b3ae692915..c3b9529fd9f 100644 --- a/tgui/packages/tgui/interfaces/ColorMatrixEditor.tsx +++ b/tgui/packages/tgui/interfaces/ColorMatrixEditor.tsx @@ -41,14 +41,15 @@ export const ColorMatrixEditor = (props) => { {`${PREFIXES[row]}${PREFIXES[col]}:`} toFixed(value, 2)} onDrag={(_, value) => { let retColor = currentColor; - retColor[row * 4 + col] = value; + retColor[row * 4 + col] = `${value}`; act('transition_color', { color: retColor, }); diff --git a/tgui/packages/tgui/interfaces/MatrixMathTester.tsx b/tgui/packages/tgui/interfaces/MatrixMathTester.tsx index 1f67fd3f3eb..6ba7bce6377 100644 --- a/tgui/packages/tgui/interfaces/MatrixMathTester.tsx +++ b/tgui/packages/tgui/interfaces/MatrixMathTester.tsx @@ -9,6 +9,8 @@ const MatrixMathTesterInput = (props: { value: number; varName: string }) => { const { act } = useBackend(); return ( toFixed(value, 3)} @@ -111,6 +113,8 @@ export const MatrixMathTester = (props) => { toFixed(value, 2)} @@ -120,6 +124,8 @@ export const MatrixMathTester = (props) => { toFixed(value, 2)} @@ -142,6 +148,8 @@ export const MatrixMathTester = (props) => { toFixed(value, 0)} @@ -151,6 +159,8 @@ export const MatrixMathTester = (props) => { toFixed(value, 0)} @@ -171,6 +181,8 @@ export const MatrixMathTester = (props) => { toFixed(value, 3)} @@ -180,6 +192,8 @@ export const MatrixMathTester = (props) => { toFixed(value, 3)} diff --git a/tgui/packages/tgui/interfaces/Mecha/ModulesPane.tsx b/tgui/packages/tgui/interfaces/Mecha/ModulesPane.tsx index 99b0fd4b7a7..f8f6af31f3e 100644 --- a/tgui/packages/tgui/interfaces/Mecha/ModulesPane.tsx +++ b/tgui/packages/tgui/interfaces/Mecha/ModulesPane.tsx @@ -547,7 +547,7 @@ const SnowflakeRadio = (props) => { { minValue={tank_pump_pressure_min} maxValue={tank_pump_pressure_max} step={10} - format={(value) => Math.round(value)} + format={(value) => `${Math.round(value)}`} onChange={(e, value) => act('equip_act', { ref: ref, diff --git a/tgui/packages/tgui/interfaces/ParticleEdit/EntriesBasic.tsx b/tgui/packages/tgui/interfaces/ParticleEdit/EntriesBasic.tsx index 3509c6d1851..7e48a4f46fb 100644 --- a/tgui/packages/tgui/interfaces/ParticleEdit/EntriesBasic.tsx +++ b/tgui/packages/tgui/interfaces/ParticleEdit/EntriesBasic.tsx @@ -40,6 +40,7 @@ export const EntryFloat = (props: EntryFloatProps) => { animated value={float} minValue={0} + maxValue={Infinity} onDrag={(e, value) => act('edit', { var: var_name, @@ -64,7 +65,9 @@ export const EntryCoord = (props: EntryCoordProps) => { /> act('edit', { var: var_name, @@ -74,7 +77,9 @@ export const EntryCoord = (props: EntryCoordProps) => { /> act('edit', { var: var_name, @@ -84,7 +89,9 @@ export const EntryCoord = (props: EntryCoordProps) => { /> act('edit', { var: var_name, @@ -274,8 +281,9 @@ export const EntryIcon = (props: EntryIconStateProps) => { act('edit', { var: var_name, @@ -363,8 +371,9 @@ export const EntryIconState = (props: EntryIconStateProps) => { act('edit', { var: var_name, diff --git a/tgui/packages/tgui/interfaces/ParticleEdit/EntriesGenerators.tsx b/tgui/packages/tgui/interfaces/ParticleEdit/EntriesGenerators.tsx index 3365b55a61c..9299228f4ae 100644 --- a/tgui/packages/tgui/interfaces/ParticleEdit/EntriesGenerators.tsx +++ b/tgui/packages/tgui/interfaces/ParticleEdit/EntriesGenerators.tsx @@ -52,7 +52,9 @@ export const FloatGenerator = (props: FloatGeneratorProps) => { act('edit', { var: var_name, @@ -177,7 +179,9 @@ export const EntryGeneratorNumbersList = ( act('edit', { var: var_name, @@ -196,6 +200,8 @@ export const EntryGeneratorNumbersList = ( act('edit', { @@ -206,6 +212,8 @@ export const EntryGeneratorNumbersList = ( /> act('edit', { @@ -217,6 +225,8 @@ export const EntryGeneratorNumbersList = ( {allow_z ? ( act('edit', { diff --git a/tgui/packages/tgui/interfaces/ParticleEdit/Generators.tsx b/tgui/packages/tgui/interfaces/ParticleEdit/Generators.tsx index c92e4b5be07..367493acc42 100644 --- a/tgui/packages/tgui/interfaces/ParticleEdit/Generators.tsx +++ b/tgui/packages/tgui/interfaces/ParticleEdit/Generators.tsx @@ -74,6 +74,7 @@ export const GeneratorListEntry = (props: GeneratorProps) => { new_value: [type, value, B, RandToNumber[rand_type]], }) } + maxValue={Infinity} /> ) : ( <> @@ -92,6 +93,8 @@ export const GeneratorListEntry = (props: GeneratorProps) => { ], }) } + minValue={-Infinity} + maxValue={Infinity} /> { new_value: [type, [A[0], value, A[2]], B, rand_type], }) } + minValue={-Infinity} + maxValue={Infinity} /> { new_value: [type, [A[0], A[1], value], B, rand_type], }) } + minValue={-Infinity} + maxValue={Infinity} /> )} @@ -131,6 +138,8 @@ export const GeneratorListEntry = (props: GeneratorProps) => { new_value: [type, A, value, RandToNumber[rand_type]], }) } + minValue={-Infinity} + maxValue={Infinity} /> ) : ( <> @@ -149,6 +158,8 @@ export const GeneratorListEntry = (props: GeneratorProps) => { ], }) } + minValue={-Infinity} + maxValue={Infinity} /> { ], }) } + minValue={-Infinity} + maxValue={Infinity} /> { ], }) } + minValue={-Infinity} + maxValue={Infinity} /> )} diff --git a/tgui/packages/tgui/interfaces/Signaler.tsx b/tgui/packages/tgui/interfaces/Signaler.tsx index 15a486a62df..a50ecae7002 100644 --- a/tgui/packages/tgui/interfaces/Signaler.tsx +++ b/tgui/packages/tgui/interfaces/Signaler.tsx @@ -35,7 +35,7 @@ export const SignalerContent = (props) => { Frequency: {