Checkbox
- Usage
- Styling
Checkbox is an input field representing a binary choice. Checkbox Group is a group of related binary choices.
new tab
<CheckboxGroup
label="Export data"
value={value}
onValueChanged={(event) => setValue(event.detail.value)}
theme="vertical"
>
<Checkbox value="0" label="Order ID" />
<Checkbox value="1" label="Product name" />
<Checkbox value="2" label="Customer" />
<Checkbox value="3" label="Status" />
</CheckboxGroup>
Use Checkbox Group to group related items. Individual checkboxes should be used for options that aren’t necessarily related to each other in any way.
States
Disabled
Disable a field to mark it as currently unavailable. Disabled state is used for fields that aren’t editable and don’t need to be readable. Disabled elements can’t be focused and may be inaccessible to assistive technologies like screen readers.
Disabling can be preferable to hiding an element to prevent changes in layout when the element’s visibility changes, and to make users aware of its existence even when currently unavailable.
new tab
<CheckboxGroup label="Departments" theme="vertical" disabled>
<Checkbox value="engineering" label="Engineering" />
<Checkbox value="human-resources" label="Human Resources" />
<Checkbox value="marketing" label="Marketing" />
<Checkbox value="operations" label="Operations" />
<Checkbox value="sales" label="Sales" />
</CheckboxGroup>
Note
|
Read-Only State
Checkbox doesn’t support read-only state.
|
Indeterminate
The indeterminate state can be used for a parent checkbox to show that there is a mix of checked and unchecked child items in a list, and to change the state of all child items at once.
new tab
function Example() {
const [items, setItems] = useState<Person[]>([]);
const [selectedIds, setSelectedIds] = useState<string[]>([]);
useEffect(() => {
getPeople({ count: 3 }).then(({ people }) => {
setItems(people);
setSelectedIds([String(people[0].id), String(people[2].id)]);
});
}, []);
return (
<VerticalLayout theme="spacing">
<Checkbox
label="Notify users"
checked={selectedIds.length === items.length}
indeterminate={selectedIds.length > 0 && selectedIds.length < items.length}
onChange={(e) => {
// TODO: This doesn't currently invoke. See https://github.com/vaadin/react-components/issues/133
setSelectedIds(e.currentTarget.checked ? items.map((person) => String(person.id)) : []);
}}
/>
<CheckboxGroup
label="Users to notify"
theme="vertical"
value={selectedIds}
onValueChanged={(event) => {
setSelectedIds(event.detail.value);
}}
>
{items.map((person) => (
<Checkbox
key={person.id}
value={String(person.id)}
label={`${person.firstName} ${person.lastName}`}
/>
))}
</CheckboxGroup>
</VerticalLayout>
);
}
Orientation
The component’s default orientation is horizontal. However, vertical orientation is recommended whenever possible as it’s easier for the user to scan a vertical list of options:
new tab
<CheckboxGroup label="Working days" theme="vertical">
<Checkbox value="mon" label="Monday" />
<Checkbox value="tue" label="Tuesday" />
<Checkbox value="wed" label="Wednesday" />
<Checkbox value="thu" label="Thursday" />
<Checkbox value="fri" label="Friday" />
<Checkbox value="sat" label="Saturday" />
<Checkbox value="sun" label="Sunday" />
</CheckboxGroup>
In cases where vertical space needs to be conserved, horizontal orientation can be used. Still, no more than three options are recommended:
new tab
<CheckboxGroup label="Permissions">
<Checkbox value="read" label="Read" />
<Checkbox value="edit" label="Edit" />
<Checkbox value="delete" label="Delete" />
</CheckboxGroup>
Basic Features
The following features, common to most input field components, are supported:
Label
The label is used to identify the input field. It supports plain-text content, and its length is limited to the width of the field. Helpers and Tooltips can be used to provide additional information that doesn’t fit into the label.
Visible labels are strongly recommended for all input fields. In cases where the built-in label cannot be used, an external element can be associated as the field’s label through the aria-labelledby
attribute. Fields without any visible label should include an invisible label for assistive technologies with the aria-label
attribute.
Helper
Helpers are used to provide additional information that the user may need to enter in the field, such as format requirements or explanations of the field’s purpose below the field.
A style variant is available for rendering the helper above the field.
In addition to plain text, helpers can contain components and HTML elements. However, complex and interactive content is likely to have accessibility issues.
Tooltip
Tooltips are small text pop-ups displayed on hover, and on keyboard-focus. They can be used to provide additional information about a field. This can be useful in situations where an always visible Helper is not appropriate. Helpers are generally recommended in favor of tooltips, though, as they provide much better discoverability and mobile support. See the Tooltip documentation for more information.
External & Invisible Labels (ARIA)
Visible labels are strongly recommended for all input fields. In situations where the built-in label cannot be used, an external element can be associated as the field’s label through its element id
. Fields without any visible label should be provided an invisible label for assistive technologies like screen readers.
<!-- Associates external element as label: -->
<label id="external-label">This is the label</label>
<vaadin-checkbox accessible-name-ref="external-label">...
<!-- Invisible label for screen readers: -->
<vaadin-checkbox accessible-name="This is the label">...
new tab
return (
<CheckboxGroup label="Label" helperText="Helper text">
<Tooltip slot="tooltip" text="Tooltip text" />
<Checkbox value="1" label="Item 1" />
<Checkbox value="2" label="Item 2" />
<Checkbox value="3" label="Item 3" />
</CheckboxGroup>
);
Style Variants
The following style variants can be applied:
Helper Above Field
The helper can be rendered above the field, and below the label.
Borders
Borders can be applied to the field surface by providing a value (e.g., 1px
) to the --vaadin-input-field-border-width
CSS property. This can be applied globally to all input fields using the html
selector, or to individual component instances. Borders are required to achieve WCAG 2.1 level AA conformant color contrast with the default Lumo styling of fields.
You can override the default border color with the --vaadin-input-field-border-color
property.
new tab
<CheckboxGroup
theme="helper-above-field"
label="Label"
helperText="Helper text"
style={{ '--vaadin-input-field-border-width': '1px' } as React.CSSProperties}
>
<Checkbox value="1" label="Item 1" />
<Checkbox value="2" label="Item 2" />
<Checkbox value="3" label="Item 3" />
</CheckboxGroup>
Best Practices
Labeling
Aim for short and descriptive labels using positive wording. Avoid negations.
It’s important to provide labels for Checkbox Groups to distinguish clearly any adjacent groups.
new tab
<VerticalLayout>
<CheckboxGroup label="Manufacturer" theme="vertical">
<Checkbox value="0" label="Akuchi" />
<Checkbox value="1" label="Broek" />
<Checkbox value="2" label="Wulf" />
</CheckboxGroup>
<CheckboxGroup label="Status" theme="vertical">
<Checkbox value="0" label="In progress" />
<Checkbox value="1" label="Done" />
<Checkbox value="2" label="Cancelled" />
</CheckboxGroup>
</VerticalLayout>
Related Components
Component | Usage Recommendation |
---|---|
A field for selecting an item from a list of options which are presented in an overlay. Recommended when there is insufficient space for a Radio Button Group. | |
A filterable, lazy loading alternative to Select. Recommended for ten or more items. | |
Scrollable list of options. Supports single and multi-select. | |
Corresponding component for mutually exclusive options, or "single-select". |