Number Input
The Number Input component provides users with a field for integer values, and buttons to increment or decrement the value.
Introduction
A number input is a UI element that accepts numeric values from the user.
Base UI's Number Input component is a customizable replacement for the native HTML <input type="number">
that solves common usability issues of the native counterpart, such as:
- Inconsistencies across browsers in the appearance and behavior of the stepper buttons
- Allowing certain non-numeric characters ('e', '+', '-', '.') and silently discarding others
- Incompatibilities with assistive technologies and limited accessibility features
See this article by the GOV.UK Design System team for a more detailed explanation.
Component
Usage
After installation, you can start building with this component using the following basic elements:
import NumberInput from '@mui/base/Unstable_NumberInput';
export default function MyApp() {
return <NumberInput />;
}
Basics
The following demo shows how to create a number input component, apply some styling, and write the latest value to a state variable using the onChange
prop:
The min
and max
props can be used to define a range of accepted values. You can pass only one of them to define an open-ended range.
<NumberInput min={-10} max={10} />
// Open-ended
<NumberInput min={0} />
The step
prop can be used to defined the granularity of the change in value when incrementing or decrementing. For example, if min={0}
and step={2}
, valid values for the component would be 0, 2, 4… since the value can only be changed in increments of 2.
// valid values: 0, 2, 4, 6, 8...
<NumberInput min={0} step={2} />
When the input field is in focus, you can enter values that fall outside the valid range. The value will be clamped based on min
, max
and step
once the input field is blurred.
Holding down the Shift key when interacting with the stepper buttons applies a multipler (default 10x) to the value change of each step.
This can be customized with the shiftMultiplier
prop. In the following snippet, if Shift is held when clicking the increment button, the value will change from 0 to 5, then to 10 etc.
<NumberInput min={0} step={1} shiftMultiplier={5} />
Here's another demo of a Number Input with fully customized styles:
Anatomy
The Base UI Number Input component consists of 4 slots:
root
: a outer<div>
containing the other interior slotsinput
: an<input>
elementincrementButton
: a<button>
for increasing the valuedecrementButton
: a<button>
for decreasing the value
<div class="MuiNumberInput-root">
<button class="MuiNumberInput-decrementButton" />
<button class="MuiNumberInput-incrementButton" />
<input class="MuiNumberInput-input" />
</div>
Slot props
Use the slots
prop to override the root slot or any interior slots:
<NumberInput
slots={{
root: 'aside',
incrementButton: CustomButton,
decrementButton: CustomButton,
}}
/>
Use the slotProps
prop to pass custom props to internal slots.
The following code snippet:
- applies a CSS class called
my-num-input
to the input slot, - and passes a
direction
prop to theCustomButton
components in the increment and decrement button slots
<NumberInput
slotProps={{
input: { className: 'my-num-input' },
incrementButton: { direction: 'UP' },
decrementButton: { direction: 'DOWN' },
}}
/>
Hook
import useNumberInput from '@mui/base/unstable_useNumberInput';
The useNumberInput
hook lets you apply the functionality of a number input to a fully custom component.
It returns props to be placed on the custom component, along with fields representing the component's internal state.
Hooks do not support slot props, but they do support customization props.
Here's an example of a custom component built using the useNumberInput
hook with all the required props:
Here's an example of a "compact" number input component using the hook that only consists of the stepper buttons.
In this demo, onChange
is used to write the latest value of the component to a state variable.
Current value: