Date Picker Combobox Example
Read This First
The code in this example is not intended for production environments.
Before using it for any purpose, read this to understand why.
This is an illustrative example of one way of using ARIA that conforms with the ARIA specification.
- There may be support gaps in some browser and assistive technology combinations, especially for mobile/touch devices. Testing code based on this example with assistive technologies is essential before considering use in production systems.
- The ARIA and Assistive Technologies Project is developing measurements of assistive technology support for APG examples.
- Robust accessibility can be further optimized by choosing implementation patterns that maximize use of semantic HTML and heeding the warning that No ARIA is better than Bad ARIA.
About This Example
The below date picker demonstrates an implementation of the Combobox Pattern that opens a dialog. The date picker dialog is opened by activating the choose date button or by moving keyboard focus to the combobox and pressing Down Arrow or Alt + Down Arrow. The dialog contains an implementation of the grid pattern for displaying a calendar and enabling selection of a date. Additional buttons in the dialog are available for changing the month and year shown in the grid.
Similar examples include:
- Date Picker Dialog example: Demonstrates a dialog containing a calendar grid for choosing a date.
- Select-Only Combobox: A single-select combobox with no text input that is functionally similar to an HTML
select
element. - Editable Combobox with Both List and Inline Autocomplete: An editable combobox that demonstrates the autocomplete behavior known as "list with inline autocomplete".
- Editable Combobox with List Autocomplete: An editable combobox that demonstrates the autocomplete behavior known as "list with manual selection".
- Editable Combobox Without Autocomplete: An editable combobox that demonstrates the behavior associated with
aria-autocomplete=none
. - Editable Combobox with Grid Popup: An editable combobox that presents suggestions in a grid, enabling users to navigate descriptive information about each suggestion.
Example
Accessibility Features
- The description of the date format is associated with the combobox via
aria-describedby
, making it available to assistive technologies as an accessible description. -
While the down arrow icon is excluded from the Tab sequence as specified in the
combobox
design pattern, it is made accessible to assistive technologies as the Choose Date button. This enables assistive technology users who might not have a keyboard, e.g., someone using a touch-based screen reader, to open the date picker dialog. - In the dialog, shortcut keys are assigned to the additional buttons for changing the month and year displayed in the calendar.
- Keyboard help is displayed at the bottom of the dialog. A live region is used to announce it to screen reader users when focus moves into the grid.
- The calendar heading displaying the month and year is marked up as a live region so screen reader users get feedback from the buttons and keyboard commands that change the month and year.
-
To facilitate compact visual design in the calendar, the day names in the column headers are abbreviated to two characters.
However, this makes it more difficult for screen reader users to understand the day names.
So, full day names are provided to assistive technologies in the HTML
abbr
attribute on the column headers, enabling screen readers to announce the full names when users navigate the grid. -
Focus and hover styling of the controls support operating system high contrast settings via the CSS border property:
- When a button or date cell receives focus a border is added.
- When hovering over a button or date cell with a pointing device a border is added.
- By default, buttons and date cells do not have a border; padding provides a placeholder space for focus and hover styling.
Keyboard Support
Combobox
Key | Function |
---|---|
Down Arrow, ALT + Down Arrow |
|
Date Picker Dialog
Key | Function |
---|---|
ESC | Closes the dialog and moves focus to the combobox. |
TAB |
|
Shift + TAB |
|
Date Picker Dialog: Calendar Buttons
Key | Function |
---|---|
Space, Enter |
Change the month and/or year displayed in the calendar grid. |
Date Picker Dialog: Date Grid
Key | Function |
---|---|
Space |
|
Enter |
|
Up Arrow | Moves focus to the same day of the previous week. |
Down Arrow | Moves focus to the same day of the next week. |
Right Arrow | Moves focus to the next day. |
Left Arrow | Moves focus to the previous day. |
Home | Moves focus to the first day (e.g. Sunday) of the current week. |
End | Moves focus to the last day (e.g. Saturday) of the current week. |
PageUp |
|
Shift+ PageUp |
|
PageDown |
|
Shift+ PageDown |
|
Date Picker Dialog: OK and Cancel Buttons
Key | Function |
---|---|
Space, Enter |
Activates the button:
|
Role, Property, State, and Tabindex Attributes
Combobox
Role | Attribute | Element | Usage |
---|---|---|---|
combobox |
input |
Identifies the input element as a combobox. |
|
aria-haspopup="dialog" |
input |
Indicates that the combobox opens a dialog. | |
aria-expanded="false" |
input |
Indicates that the combobox is collapsed, i.e., the "Choose Date" dialog is not displayed. | |
aria-expanded="true" |
input |
Indicates that the combobox is expanded, i.e., the "Choose Date" dialog is open. | |
aria-autocomplete="none" |
input |
Indicates the combobox does not support autocomplete. | |
aria-controls="IDREF" |
input |
Identifies the element controlled by the combobox. | |
aria-describedby="IDREF" |
input |
Identifies the element that provides an accessible description for the combobox, enabling assistive technologies to associate the date format description with the input. |
Choose Date Button
Role | Attribute | Element | Usage |
---|---|---|---|
tabindex="-1" |
button |
Excludes the button (i.e., the down arrow icon) from the Tab sequence as specified by the combobox design pattern. | |
aria-label="string" |
button |
Defines the accessible name as "Choose Date", which matches the title of the dialog opened by activating the button. |
Date Picker Dialog
Role | Attribute | Element | Usage |
---|---|---|---|
dialog |
div
|
Identifies the element as a dialog. | |
aria-modal="true" |
div |
Indicates the dialog is modal. | |
aria-label="Choose Date" |
div |
Defines the accessible name for the dialog. | |
aria-live="polite" |
h2 |
|
|
aria-live="polite" |
div |
|
Date Picker Dialog: Calendar Navigation Buttons
Role | Attribute | Element | Usage |
---|---|---|---|
aria-label="String" |
button |
Defines the accessible name of the button (e.g. "Next Year"). |
Date Picker Dialog: Date Grid
Role | Attribute | Element | Usage |
---|---|---|---|
grid |
table
|
|
|
aria-labelledby="IDREF" |
table |
Identifies the element that provides the accessible name for the grid , which is the h2 that shows the month and year of the dates displayed in the grid. |
|
tabindex="0"
|
td
|
|
|
tabindex="-1"
|
td
|
|
|
aria-selected="true" |
td |
|
Note: Since the names of the days of the week in the column headers are abbreviated to two characters, they may be difficult to understand when announced by a screen reader.
An alternative column header name can be provided to screen readers by applying the abbr
attribute to the th
elements.
So, each th
element includes an abbr
attribute containing the full spelling of the name of the day for that column.
Javascript and CSS Source Code
- CSS: combobox-datepicker.css
- Javascript: combobox-datepicker.js
HTML Source Code
<div id="myDatepicker" class="combobox-datepicker">
<div class="label">
<label id="id-label-1" for="cb-textbox-1">
Date
</label>
</div>
<div class="group">
<input type="text"
id="cb-textbox-1"
aria-autocomplete="none"
role="combobox"
aria-expanded="false"
aria-haspopup="dialog"
aria-controls="cb-dialog-1"
aria-describedby="cb-description-1">
<span class="desc" id="cb-description-1">
(date format: mm/dd/yyyy)
</span>
<button type="button"
class="arrow"
tabindex="-1"
aria-label="Choose Date">
<span>
<svg width="18"
height="16"
aria-hidden="true"
focusable="false">
<polygon points="2,4 16,4 9,14"></polygon>
</svg>
</span>
</button>
</div>
<div id="cb-dialog-1"
class="dialog"
role="dialog"
aria-modal="true"
aria-label="Choose Date">
<div class="header">
<button type="button"
class="prev-year"
aria-label="previous year">
<span class="fas fa-angle-double-left fa-lg"></span>
</button>
<button type="button"
class="prev-month"
aria-label="previous month">
<span class="fas fa-angle-left fa-lg"></span>
</button>
<h2 id="cb-grid-label"
class="month-year"
aria-live="polite">
December 2020
</h2>
<button type="button"
class="next-month"
aria-label="next month">
<span class="fas fa-angle-right fa-lg"></span>
</button>
<button type="button"
class="next-year"
aria-label="next year">
<span class="fas fa-angle-double-right fa-lg"></span>
</button>
</div>
<div class="table-wrap">
<table class="dates"
role="grid"
aria-labelledby="cb-grid-label">
<thead>
<tr>
<th scope="col" abbr="Sunday">
Su
</th>
<th scope="col" abbr="Monday">
Mo
</th>
<th scope="col" abbr="Tuesday">
Tu
</th>
<th scope="col" abbr="Wednesday">
We
</th>
<th scope="col" abbr="Thursday">
Th
</th>
<th scope="col" abbr="Friday">
Fr
</th>
<th scope="col" abbr="Saturday">
Sa
</th>
</tr>
</thead>
<tbody>
<tr>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td tabindex="-1" data-date="2020-02-01">
1
</td>
</tr>
<tr>
<td tabindex="-1" data-date="2020-02-02">
2
</td>
<td tabindex="-1" data-date="2020-02-03">
3
</td>
<td tabindex="-1" data-date="2020-02-04">
4
</td>
<td tabindex="-1" data-date="2020-02-05">
5
</td>
<td tabindex="-1" data-date="2020-02-06">
6
</td>
<td tabindex="-1" data-date="2020-02-07">
7
</td>
<td tabindex="-1" data-date="2020-02-08">
8
</td>
</tr>
<tr>
<td tabindex="-1" data-date="2020-02-09">
9
</td>
<td tabindex="-1" data-date="2020-02-10">
10
</td>
<td tabindex="-1" data-date="2020-02-11">
11
</td>
<td tabindex="-1" data-date="2020-02-12">
12
</td>
<td tabindex="-1" data-date="2020-02-13">
13
</td>
<td role="gridcell"
aria-selected="true"
data-date="2020-02-14">
14
</td>
<td tabindex="-1" data-date="2020-02-15">
15
</td>
</tr>
<tr>
<td tabindex="-1" data-date="2020-02-16">
16
</td>
<td tabindex="-1" data-date="2020-02-17">
17
</td>
<td tabindex="-1" data-date="2020-02-18">
18
</td>
<td tabindex="-1" data-date="2020-02-19">
19
</td>
<td tabindex="-1" data-date="2020-02-20">
20
</td>
<td tabindex="-1" data-date="2020-02-21">
21
</td>
<td tabindex="-1" data-date="2020-02-22">
22
</td>
</tr>
<tr>
<td tabindex="-1" data-date="2020-02-23">
23
</td>
<td tabindex="-1" data-date="2020-02-24">
24
</td>
<td tabindex="-1" data-date="2020-02-25">
25
</td>
<td tabindex="-1" data-date="2020-02-26">
26
</td>
<td tabindex="-1" data-date="2020-02-27">
27
</td>
<td tabindex="-1" data-date="2020-02-28">
28
</td>
<td tabindex="-1" data-date="2020-02-29">
29
</td>
</tr>
<tr>
<td tabindex="-1" data-date="2020-02-30">
30
</td>
<td tabindex="-1" data-date="2020-02-31">
31
</td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
<td class="disabled" tabindex="-1"></td>
</tr>
</tbody>
</table>
</div>
<div class="dialog-ok-cancel-group">
<button class="dialog-button" value="cancel">
Cancel
</button>
<button class="dialog-button" value="ok">
OK
</button>
</div>
<div class="dialog-message" aria-live="polite"></div>
</div>
</div>