2022-10-03 16:15:26 +00:00
|
|
|
import {
|
|
|
|
FormControl,
|
|
|
|
NumberInput,
|
|
|
|
NumberInputField,
|
|
|
|
NumberIncrementStepper,
|
|
|
|
NumberDecrementStepper,
|
|
|
|
NumberInputProps,
|
|
|
|
FormLabel,
|
2022-10-27 04:24:00 +00:00
|
|
|
NumberInputFieldProps,
|
|
|
|
NumberInputStepperProps,
|
|
|
|
FormControlProps,
|
|
|
|
FormLabelProps,
|
|
|
|
TooltipProps,
|
|
|
|
Tooltip,
|
2022-10-03 16:15:26 +00:00
|
|
|
} from '@chakra-ui/react';
|
|
|
|
import _ from 'lodash';
|
|
|
|
import { FocusEvent, useEffect, useState } from 'react';
|
|
|
|
|
|
|
|
const numberStringRegex = /^-?(0\.)?\.?$/;
|
|
|
|
|
|
|
|
interface Props extends Omit<NumberInputProps, 'onChange'> {
|
|
|
|
styleClass?: string;
|
|
|
|
label?: string;
|
|
|
|
width?: string | number;
|
|
|
|
showStepper?: boolean;
|
|
|
|
value: number;
|
|
|
|
onChange: (v: number) => void;
|
|
|
|
min: number;
|
|
|
|
max: number;
|
|
|
|
clamp?: boolean;
|
|
|
|
isInteger?: boolean;
|
2022-10-27 04:24:00 +00:00
|
|
|
formControlProps?: FormControlProps;
|
|
|
|
formLabelProps?: FormLabelProps;
|
|
|
|
numberInputProps?: NumberInputProps;
|
|
|
|
numberInputFieldProps?: NumberInputFieldProps;
|
|
|
|
numberInputStepperProps?: NumberInputStepperProps;
|
|
|
|
tooltipProps?: Omit<TooltipProps, 'children'>;
|
2022-10-03 16:15:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Customized Chakra FormControl + NumberInput multi-part component.
|
|
|
|
*/
|
|
|
|
const IAINumberInput = (props: Props) => {
|
|
|
|
const {
|
|
|
|
label,
|
|
|
|
styleClass,
|
|
|
|
isDisabled = false,
|
|
|
|
showStepper = true,
|
|
|
|
width,
|
|
|
|
textAlign,
|
|
|
|
isInvalid,
|
|
|
|
value,
|
|
|
|
onChange,
|
|
|
|
min,
|
|
|
|
max,
|
|
|
|
isInteger = true,
|
2022-10-27 04:24:00 +00:00
|
|
|
formControlProps,
|
|
|
|
formLabelProps,
|
|
|
|
numberInputFieldProps,
|
|
|
|
numberInputStepperProps,
|
|
|
|
tooltipProps,
|
2022-10-03 16:15:26 +00:00
|
|
|
...rest
|
|
|
|
} = props;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
* 5, the value has been parsed from "1." to "1" and they end up with "15".
|
|
|
|
*
|
|
|
|
* To resolve this, this component keeps a the value as a string internally,
|
|
|
|
* and the UI component uses that. When a change is made, that string is parsed
|
|
|
|
* as a number and given to the `onChange` function.
|
|
|
|
*/
|
|
|
|
|
|
|
|
const [valueAsString, setValueAsString] = useState<string>(String(value));
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When `value` changes (e.g. from a diff source than this component), we need
|
|
|
|
* to update the internal `valueAsString`, but only if the actual value is different
|
|
|
|
* from the current value.
|
|
|
|
*/
|
|
|
|
useEffect(() => {
|
2022-10-27 04:24:00 +00:00
|
|
|
if (
|
|
|
|
!valueAsString.match(numberStringRegex) &&
|
|
|
|
value !== Number(valueAsString)
|
|
|
|
) {
|
2022-10-03 16:15:26 +00:00
|
|
|
setValueAsString(String(value));
|
|
|
|
}
|
|
|
|
}, [value, valueAsString]);
|
|
|
|
|
|
|
|
const handleOnChange = (v: string) => {
|
|
|
|
setValueAsString(v);
|
|
|
|
// This allows negatives and decimals e.g. '-123', `.5`, `-0.2`, etc.
|
|
|
|
if (!v.match(numberStringRegex)) {
|
|
|
|
// Cast the value to number. Floor it if it should be an integer.
|
|
|
|
onChange(isInteger ? Math.floor(Number(v)) : Number(v));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clicking the steppers allows the value to go outside bounds; we need to
|
|
|
|
* clamp it on blur and floor it if needed.
|
|
|
|
*/
|
|
|
|
const handleBlur = (e: FocusEvent<HTMLInputElement>) => {
|
|
|
|
const clamped = _.clamp(
|
|
|
|
isInteger ? Math.floor(Number(e.target.value)) : Number(e.target.value),
|
|
|
|
min,
|
|
|
|
max
|
|
|
|
);
|
|
|
|
setValueAsString(String(clamped));
|
|
|
|
onChange(clamped);
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
2022-10-27 04:24:00 +00:00
|
|
|
<Tooltip {...tooltipProps}>
|
|
|
|
<FormControl
|
|
|
|
isDisabled={isDisabled}
|
|
|
|
isInvalid={isInvalid}
|
|
|
|
className={`invokeai__number-input-form-control ${styleClass}`}
|
|
|
|
{...formControlProps}
|
|
|
|
>
|
2022-10-03 16:15:26 +00:00
|
|
|
<FormLabel
|
2022-10-27 04:24:00 +00:00
|
|
|
className="invokeai__number-input-form-label"
|
|
|
|
style={{ display: label ? 'block' : 'none' }}
|
|
|
|
{...formLabelProps}
|
2022-10-03 16:15:26 +00:00
|
|
|
>
|
|
|
|
{label}
|
|
|
|
</FormLabel>
|
2022-10-27 04:24:00 +00:00
|
|
|
<NumberInput
|
|
|
|
className="invokeai__number-input-root"
|
|
|
|
value={valueAsString}
|
|
|
|
keepWithinRange={true}
|
|
|
|
clampValueOnBlur={false}
|
|
|
|
onChange={handleOnChange}
|
|
|
|
onBlur={handleBlur}
|
2022-10-03 16:15:26 +00:00
|
|
|
width={width}
|
2022-10-27 04:24:00 +00:00
|
|
|
{...rest}
|
2022-10-03 16:15:26 +00:00
|
|
|
>
|
2022-10-27 04:24:00 +00:00
|
|
|
<NumberInputField
|
|
|
|
className="invokeai__number-input-field"
|
|
|
|
textAlign={textAlign}
|
|
|
|
{...numberInputFieldProps}
|
|
|
|
/>
|
|
|
|
<div
|
|
|
|
className="invokeai__number-input-stepper"
|
|
|
|
style={showStepper ? { display: 'block' } : { display: 'none' }}
|
|
|
|
>
|
|
|
|
<NumberIncrementStepper
|
|
|
|
{...numberInputStepperProps}
|
|
|
|
className="invokeai__number-input-stepper-button"
|
|
|
|
/>
|
|
|
|
<NumberDecrementStepper
|
|
|
|
{...numberInputStepperProps}
|
|
|
|
className="invokeai__number-input-stepper-button"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</NumberInput>
|
|
|
|
</FormControl>
|
|
|
|
</Tooltip>
|
2022-10-03 16:15:26 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default IAINumberInput;
|