Must haves
- Programmatically labeled via
<label>
withfor/id
,aria-label
, oraria-labelledby
- Persistently visually labeled - don’t rely on
placeholder
that disappears when text is entered. - Any field constraints are clearly communicated and programmatically associated with the field (either included in the label or associated with
aria-describedby
) - Groups of related fields are contained in a
<fieldset>
with an associated<legend>
- If an error state can be triggered, and inline error messages are added, they are programmatically associated with the field using
aria-describedby
- If an error state can be triggered, focus moves to the first field in error (or if an error summary is added at the top of the form, move focus there).
Learn more about creating accessible forms
Visual example
Text field | Error state |
---|---|
![]() | ![]() |
Do and Don’t
Do | Don't |
---|---|
|
Here the text inputs are only labeled by The checkboxes are not programmatically labeled because they are not associated with the The checkboxes are not correctly grouped in a |