eo-n/ui
UIComponentsNumber Input

Number Input

Adjusts numerical values via buttons, text entry, or dragging.

Installation

CLI

npx shadcn@latest add "https://eo-n.vercel.app/r/number-input"

Manual

Copy and paste the following code into your project.

import * as React from "react";
import { NumberField as NumberInputPrimitive } from "@base-ui-components/react/number-field";
import { Minus, MoveHorizontal, MoveVertical, Plus } from "lucide-react";
 
import { cn } from "@/lib/utils";
 
function NumberInput({
  className,
  ...props
}: React.ComponentProps<typeof NumberInputPrimitive.Root>) {
  return (
    <NumberInputPrimitive.Root
      data-slot="number-input"
      className={cn("flex w-full flex-col items-start gap-0.5", className)}
      {...props}
    />
  );
}
 
function NumberInputScrubArea({
  className,
  children,
  direction,
  ...props
}: React.ComponentProps<typeof NumberInputPrimitive.ScrubArea>) {
  return (
    <NumberInputPrimitive.ScrubArea
      data-slot="number-input-scrub-area"
      className={cn(
        direction === "horizontal" ? "cursor-ew-resize" : "cursor-ns-resize",
        className
      )}
      direction={direction}
      {...props}
    >
      {children}
      <NumberInputPrimitive.ScrubAreaCursor data-slot="number-input-scrub-area-cursor">
        {direction === "vertical" ? (
          <MoveVertical className="size-5" />
        ) : (
          <MoveHorizontal className="size-5" />
        )}
      </NumberInputPrimitive.ScrubAreaCursor>
    </NumberInputPrimitive.ScrubArea>
  );
}
 
function NumberInputGroup({
  className,
  ...props
}: React.ComponentProps<typeof NumberInputPrimitive.Group>) {
  return (
    <NumberInputPrimitive.Group
      data-slot="number-input-group"
      className={cn("flex h-9 w-full items-center", className)}
      {...props}
    />
  );
}
 
function NumberInputDecrement({
  className,
  ...props
}: React.ComponentProps<typeof NumberInputPrimitive.Decrement>) {
  return (
    <NumberInputPrimitive.Decrement
      data-slot="number-input-decrement"
      className={cn(
        "bg-background hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50 inline-flex size-9 shrink-0 cursor-pointer items-center justify-center rounded-l-md border border-r-0 shadow-xs transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
        className
      )}
      {...props}
    >
      <Minus />
    </NumberInputPrimitive.Decrement>
  );
}
 
function NumberInputField({
  className,
  ...props
}: React.ComponentProps<typeof NumberInputPrimitive.Input>) {
  return (
    <NumberInputPrimitive.Input
      data-slot="number-input-field"
      className={cn(
        "border-input placeholder:text-muted-foreground selection:bg-primary selection:text-primary-foreground dark:bg-input/30 flex h-full w-full min-w-0 border bg-transparent px-3 py-1 text-center text-base tabular-nums transition-[color,box-shadow] ease-out outline-none data-disabled:pointer-events-none data-disabled:opacity-50 md:text-sm",
        "focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px]",
        "aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
        className
      )}
      {...props}
    />
  );
}
 
function NumberInputIncrement({
  className,
  ...props
}: React.ComponentProps<typeof NumberInputPrimitive.Increment>) {
  return (
    <NumberInputPrimitive.Increment
      data-slot="number-input-increment"
      className={cn(
        "bg-background hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50 inline-flex size-9 shrink-0 cursor-pointer items-center justify-center rounded-r-md border border-l-0 shadow-xs transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
        className
      )}
      {...props}
    >
      <Plus />
    </NumberInputPrimitive.Increment>
  );
}
 
export {
  NumberInput,
  NumberInputScrubArea,
  NumberInputGroup,
  NumberInputDecrement,
  NumberInputField,
  NumberInputIncrement,
};
Update the import paths to match your project setup.

Usage

Import all parts and piece them together.

import {
  NumberInput,
  NumberInputDecrement,
  NumberInputField,
  NumberInputGroup,
  NumberInputIncrement,
  NumberInputScrubArea,
  NumberInputScrubAreaCursor,
} from "@/components/ui/number-input";
<NumberInput>
  <NumberInputScrubArea>
    <Label>Age</Label>
    <NumberInputScrubAreaCursor />
  </NumberInputScrubArea>
  <NumberInputGroup>
    <NumberInputDecrement />
    <NumberInputField />
    <NumberInputIncrement />
  </NumberInputGroup>
</NumberInput>

Examples

Bounded Number Input

Disabled

Scrub Direction

On this page