AniUI

Stepper

Numeric increment/decrement control with min, max, and step support. Ideal for quantity selectors.

1
Web preview — components render natively on iOS & Android
import { Stepper } from "@/components/ui/stepper";

const [count, setCount] = useState(1);
<Stepper value={count} onChange={setCount} min={0} max={10} />

Installation#

npx @aniui/cli add stepper

Sizes#

1
1
1
Web preview — components render natively on iOS & Android
<Stepper size="sm" value={1} onChange={setCount} />
<Stepper size="md" value={1} onChange={setCount} />
<Stepper size="lg" value={1} onChange={setCount} />

Custom Step#

Use the step prop to control the increment/decrement amount.

0
Step: 5
Web preview — components render natively on iOS & Android
<Stepper value={0} onChange={setCount} min={0} max={100} step={5} />

Props#

PropTypeDefault
value
number
onChange
(value: number) => void
min
number
0
max
number
99
step
number
1
size
"sm" | "md" | "lg"
"md"
className
string

Accessibility#

  • accessibilityRole="adjustable" with increment/decrement buttons.
  • Current value is announced to screen readers via accessibilityValue.

Source#

components/ui/stepper.tsx
import React from "react";
import { View, Text, Pressable } from "react-native";

const sizes = {
  sm: { height: 36, width: 36 },
  md: { height: 44, width: 44 },
  lg: { height: 56, width: 56 },
} as const;
export interface StepperProps extends React.ComponentPropsWithoutRef<typeof View> {
  className?: string;
  value: number;
  onChange: (value: number) => void;
  min?: number;
  max?: number;
  step?: number;
  size?: "sm" | "md" | "lg";
}
export function Stepper({ size = "md", className, value, onChange, min = 0, max = 99, step = 1, ...props }: StepperProps) {
  const s = sizes[size];
  const canDec = value - step >= min;
  const canInc = value + step <= max;
  return (
    <View
      style={{ flexDirection: "row", alignItems: "center", height: s.height, borderWidth: 1, borderColor: "#e4e4e7", borderRadius: 8, alignSelf: "flex-start" }}
      accessibilityRole="adjustable" accessibilityValue={{ min, max, now: value }}
      {...props}
    >
      <Pressable
        style={{ width: s.width, height: "100%", alignItems: "center", justifyContent: "center", borderRightWidth: 1, borderRightColor: "#e4e4e7", opacity: canDec ? 1 : 0.3 }}
        onPress={() => { if (canDec) onChange(value - step); }}
        disabled={!canDec} accessible={true} accessibilityRole="button" accessibilityLabel="Decrease"
      >
        <Text style={{ fontSize: 18, color: "#09090b" }}>−</Text>
      </Pressable>
      <View style={{ width: 56, alignItems: "center", justifyContent: "center" }}>
        <Text style={{ fontSize: 16, fontWeight: "500", color: "#09090b" }}>{value}</Text>
      </View>
      <Pressable
        style={{ width: s.width, height: "100%", alignItems: "center", justifyContent: "center", borderLeftWidth: 1, borderLeftColor: "#e4e4e7", opacity: canInc ? 1 : 0.3 }}
        onPress={() => { if (canInc) onChange(value + step); }}
        disabled={!canInc} accessible={true} accessibilityRole="button" accessibilityLabel="Increase"
      >
        <Text style={{ fontSize: 18, color: "#09090b" }}>+</Text>
      </Pressable>
    </View>
  );
}