Checkbox Input
A checkbox input is a minimal headless wrapper around the native HTML <input type="checkbox"> element. Unlike a Checkbox component which includes a wrapping <label>, CheckboxInput uses aria-label for accessibility, giving consumers full control over label placement and visual presentation.
This component is ideal when you need a bare checkbox that you will wrap in your own custom layout, such as placing the label to the left, using a custom indicator, or integrating into a complex form where label positioning must be controlled by the consumer.
Implementation Notes
- Renders a single
<input type="checkbox">element with no wrapper - Uses
aria-labelfor accessible naming instead of a wrapping<label>element - Supports two-way binding on the
checkedprop - Spreads
restPropsonto the input element for consumer extensibility
Props
checked: boolean (default: false) -- bindable boolean indicating whether the checkbox is checkedlabel: string (required) -- accessible name applied viaaria-labeldisabled: boolean (default: false) -- whether the checkbox is disabledrequired: boolean (default: false) -- whether the checkbox is required for form submissionname: string (default: undefined) -- form field name attributeid: string (default: undefined) -- element id attributevalue: string (default: undefined) -- value attribute for form submission...restProps: unknown -- additional attributes spread onto the input element
Usage
<Fieldset legend="What symptoms are you experiencing?">
<Hint>Select all that apply</Hint>
<CheckboxGroup>
<CheckboxInput label="Headache" name="symptoms" value="headache" />
<CheckboxInput label="Fever" name="symptoms" value="fever" />
<CheckboxInput label="Cough" name="symptoms" value="cough" />
<CheckboxInput label="Fatigue" name="symptoms" value="fatigue" />
</CheckboxGroup>
</Fieldset>
<CheckboxInput label="I agree to the terms and conditions" checked={agreed} required name="terms" value="yes" />
<CheckboxInput label="Subscribe to newsletter" checked={subscribed} disabled={!hasEmail} />
Keyboard Interactions
- Space: Toggles the checkbox checked state (native browser behavior)
- Tab: Moves focus to and from the checkbox (native browser behavior)
ARIA
aria-label={label}-- provides an accessible name for the checkbox since there is no visible<label>element
When to Use
- Use when users need to select one or more options from a list of choices
- Use to toggle a single boolean option on or off, such as agreeing to terms or opting into communications
- Use within a Fieldset with a legend that describes the group as a whole
- Use with Hint text to provide guidance like "Select all options that apply to you"
When Not to Use
- Do not use when users can only select one option — use RadioInput within a RadioGroup instead
- Do not use for toggling a visible on/off setting — use SwitchButton for immediate-effect toggles
- Do not pre-select checkbox options — users may miss questions or submit wrong answers
- Do not use outside of a CheckboxGroup when managing shared state across multiple checkboxes
Headless
This headless CheckboxInput component provides a native <input type="checkbox"> with aria-label for accessible naming, two-way binding on the checked prop, and support for disabled, required, and form attributes. The consumer provides all visual styling including custom checkbox indicators, colors, sizes, and label positioning.
Styles
The consumer provides all CSS styling. The component renders with a .checkbox-input class for targeting. No default styles are included — this is a fully headless component.
Testing
- Verify the component renders a
<input>element with classcheckbox-input - Verify aria-label={label}
-- provides an accessible name for the checkbox since there is no visible<label>` element - Verify keyboard interactions work correctly
- Verify pass-through attributes are applied
Advice
- Designers: Ensure custom checkbox indicators have sufficient size (at least 24x24px touch target) and clear checked/unchecked states with adequate contrast.
- Developers: Use the
aria-labelprop to provide an accessible name. For visible labels, wrap the CheckboxInput in your own<label>element or use a separate linked label.
Related components
checkbox-group— a group component that manages a collection of checkboxes with shared stateradio-input— a single radio button input <input type="radio">switch-button— a toggle switch for turning a setting on or offtoggle-button— a button that toggles between pressed and unpressed states
References
- MDN checkbox input: https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/checkbox
- WAI-ARIA Checkbox Pattern: https://www.w3.org/WAI/ARIA/apg/patterns/checkbox/