WAI-ARIA basics
Following on from the previous article, sometimes making complex UI controls that involve unsemantic HTML and dynamic JavaScript-updated content can be difficult. WAI-ARIA is a technology that can help with such problems by adding in further semantics that browsers and assistive technologies can recognize and use to let users know what is going on. Here we'll show how to use it at a basic level to improve accessibility.
| Prerequisites: | Familiarity with HTML, CSS, and accessibility best practices as taught in previous lessons in the module. | 
|---|---|
| Learning outcomes: | 
 | 
What is WAI-ARIA?
Let's start by looking at what WAI-ARIA is, and what it can do for us.
A whole new set of problems
As web apps started to get more complex and dynamic, a new set of accessibility features and problems started to appear.
For example, HTML introduced a number of semantic elements to define common page features (<nav>, <footer>, etc.). Before these were available, developers would use <div>s with IDs or classes, e.g., <div class="nav">, but these were problematic, as there was no easy way to easily find a specific page feature such as the main navigation programmatically.
The initial solution was to add one or more hidden links at the top of the page to link to the navigation (or whatever else), for example:
<a href="#hidden" class="hidden">Skip to navigation</a>
But this is still not very precise, and can only be used when the screen reader is reading from the top of the page.
As another example, apps started to feature complex controls like date pickers for choosing dates, sliders for choosing values, etc. HTML provides special input types to render such controls:
<input type="date" /> <input type="range" />
These were originally not well-supported and it was, and still is to a lesser extent, difficult to style them, leading designers and developers to opt for custom solutions. Instead of using these native features, some developers rely on JavaScript libraries that generate such controls as a series of nested <div>s which are then styled using CSS and controlled using JavaScript.
The problem here is that visually they work, but screen readers can't make any sense of what they are at all, and their users just get told that they can see a jumble of elements with no semantics to describe what they mean.
Enter WAI-ARIA
WAI-ARIA (Web Accessibility Initiative - Accessible Rich Internet Applications) is a specification written by the W3C, defining a set of additional HTML attributes that can be applied to elements to provide additional semantics and improve accessibility wherever it is lacking. There are three main features defined in the spec:
- Roles
- 
These define what an element is or does. Many of these are so-called landmark roles, which largely duplicate the semantic value of structural elements, such as role="navigation"(<nav>),role="banner"(document<header>),role="complementary"(<aside>) or ,role="search"(<search>). Some other roles describe different page structures that do not have elements with that match those roles, such asrole="tablist", androle="tabpanel", which are commonly found in UIs.
- Properties
- 
These define properties of elements, which can be used to give them extra meaning or semantics. As an example, aria-required="true"specifies that a form input needs to be filled in order to be valid, whereasaria-labelledby="label"allows you to put an ID on an element, then reference it as being the label for anything else on the page, including multiple elements, which is not possible using<label for="input">. As an example, you could usearia-labelledbyto specify that a key description contained in a<div>is the label for multiple table cells, or you could use it as an alternative to image alt text — specify existing information on the page as an image's alt text, rather than having to repeat it inside thealtattribute. You can see an example of this at Text alternatives.
- States
- 
Special properties that define the current conditions of elements, such as aria-disabled="true", which specifies to a screen reader that a form input is currently disabled. States differ from properties in that properties don't change throughout the lifecycle of an app, whereas states can change, generally programmatically via JavaScript.
An important point about WAI-ARIA attributes is that they don't affect anything about the web page, except for the information exposed by the browser's accessibility APIs (where screen readers get their information from). WAI-ARIA doesn't affect webpage structure, the DOM, etc., although the attributes can be useful for selecting elements by CSS.
Note: You can find a useful list of all the ARIA roles and their uses, with links to further information, in the WAI-ARIA spec — see Definition of Roles — on this site — see ARIA roles.
The spec also contains a list of all the properties and states, with links to further information — see Definitions of States and Properties (all aria-* attributes).
Where is WAI-ARIA supported?
This is not an easy question to answer. It is difficult to find a conclusive resource that states what features of WAI-ARIA are supported, and where, because:
- There are a lot of features in the WAI-ARIA spec.
- There are many combinations of operating systems, browsers, and screen readers to consider.
This last point is key — To use a screen reader in the first place, your operating system needs to run browsers that have the necessary accessibility APIs in place to expose the information screen readers need to do their job. Most popular OSes have one or two browsers in place that screen readers can work with. The Paciello Group has a fairly up-to-date post that provides data for this — see Rough Guide: browsers, operating systems and screen reader support updated.
Next, you need to worry about whether the browsers in question support ARIA features and expose them via their APIs, but also whether screen readers recognize that information and present it to their users in a useful way.
- Browser support is almost universal.
- Screen reader support for ARIA features isn't quite at this level, but the most popular screen readers are getting there. You can get an idea of support levels by looking at Powermapper's WAI-ARIA Screen reader compatibility article.
In this article, we won't attempt to cover every WAI-ARIA feature, and its exact support details. Instead, we will cover the most critical WAI-ARIA features for you to know about; if we don't mention any support details, you can assume that the feature is well-supported. We will clearly mention any exceptions to this.
Note: Some JavaScript libraries support WAI-ARIA, meaning that when they generate UI features like complex form controls, they add ARIA attributes to improve the accessibility of those features. If you are looking for a 3rd party JavaScript solution for rapid UI development, you should definitely consider the accessibility of its UI widgets as an important factor when making your choice. Good examples are jQuery UI (see About jQuery UI: Deep accessibility support), ExtJS, and Dojo/Dijit.
When should you use WAI-ARIA?
We talked about some of the problems that prompted WAI-ARIA to be created earlier on, but essentially, there are four main areas that WAI-ARIA is useful in:
- Signposts/Landmarks
- 
ARIA's roleattribute values can act as landmarks that either replicate the semantics of HTML elements (e.g.,<nav>), or go beyond HTML semantics to provide signposts to different functional areas, for example,search,tablist,tab,listbox, etc.
- Dynamic content updates
- 
Screen readers tend to have difficulty with reporting constantly changing content; with ARIA we can use aria-liveto inform screen reader users when an area of content is updated dynamically: for example, by JavaScript in the page fetching new content from the server and updating the DOM.
- Enhancing keyboard accessibility
- 
There are built-in HTML elements that have native keyboard accessibility; when other elements are used along with JavaScript to simulate similar interactions, keyboard accessibility and screen reader reporting suffers as a result. Where this is unavoidable, WAI-ARIA provides a means to allow other elements to receive focus (using tabindex).
- Accessibility of non-semantic controls
- 
When a series of nested <div>s along with CSS/JavaScript is used to create a complex UI-feature, or a native control is greatly enhanced/changed via JavaScript, accessibility can suffer — screen reader users will find it difficult to work out what the feature does if there are no semantics or other clues. In these situations, ARIA can help to provide what's missing with a combination of roles likebutton,listbox, ortablist, and properties likearia-requiredoraria-posinsetto provide further clues as to functionality.
In the next section, we'll look at the four main areas described earlier in more detail, along with examples. Before you continue, you should put a screen reader testing setup in place, so you can test some of the examples as you go through. See our section on testing screen readers for more information.
You should only use WAI-ARIA when you need to!
Using the correct HTML elements implicitly gives you the roles that are needed and you should always use native HTML features to provide the semantics required by screen readers to tell their users what is going on. Sometimes this isn't possible, either because you have limited control over the code, or because you are creating something complex that doesn't have an easy HTML element to implement it. In such cases, WAI-ARIA can be a valuable accessibility enhancing tool.
But again, only use it when necessary!
Also, try to make sure you test your site with a variety of real users — non-disabled people, people using screen readers, people using keyboard navigation, etc. They will have better insights than you about how well it works.
Signposts/Landmarks
WAI-ARIA adds the role attribute to browsers, which allows you to add extra semantic value to elements on your site wherever they are needed. The first major area in which this is useful is providing information for screen readers so that their users can find common page elements. This example has the following structure:
<header>
  <h1>Header</h1>
  <!-- Even is it's not mandatory, it's common practice to put the main navigation menu within the main header -->
  <nav>
    <ul>
      <li><a href="#">Home</a></li>
      <li><a href="#">Team</a></li>
      <li><a href="#">Projects</a></li>
      <li><a href="#">Contact</a></li>
    </ul>
    <!-- A Search form is another common non-linear way to navigate through a website. -->
    <form>
      <input type="search" name="q" placeholder="Search query" />
      <input type="submit" value="Go!" />
    </form>
  </nav>
</header>
<!-- Here is our page's main content -->
<main>
  <!-- It contains an article -->
  <article>
    <h2>Article heading</h2>
    <p>
      Lorem ipsum dolor sit amet, consectetur adipisicing elit. Donec a diam
      lectus. Set sit amet ipsum mauris. Maecenas congue ligula as quam viverra
      nec consectetur ant hendrerit. Donec et mollis dolor. Praesent et diam
      eget libero egestas mattis sit amet vitae augue. Nam tincidunt congue
      enim, ut porta lorem lacinia consectetur.
    </p>
    <h3>subsection</h3>
    <p>
      Donec ut librero sed accu vehicula ultricies a non tortor. Lorem ipsum
      dolor sit amet, consectetur adipisicing elit. Aenean ut gravida lorem. Ut
      turpis felis, pulvinar a semper sed, adipiscing id dolor.
    </p>
  </article>
  <!-- the aside content can also be nested within the main content -->
  <aside>
    <h2>Related</h2>
    <ul>
      <li><a href="#">Oh I do like to be beside the seaside</a></li>
      <li><a href="#">Oh I do like to be beside the sea</a></li>
      <li><a href="#">Although in the North of England</a></li>
      <li><a href="#">It never stops raining</a></li>
      <li><a href="#">Oh well...</a></li>
    </ul>
  </aside>
</main>
<!-- And here is our main footer that is used across all the pages of our website -->
<footer>
  <p>©Copyright 2050 by nobody. All rights reversed.</p>
</footer>
If you try testing the example with a screen reader in a modern browser, you'll already get some useful information. For example, VoiceOver gives you the following:
- On the <header>element — "banner, 2 items" (it contains a heading and the<nav>).
- On the <nav>element — "navigation 2 items" (it contains a list and a form).
- On the <main>element — "main 2 items" (it contains an article and an aside).
- On the <aside>element — "complementary 2 items" (it contains a heading and a list).
- On the search form input — "Search query, insertion at beginning of text".
- On the <footer>element — "footer 1 item".
If you go to VoiceOver's landmarks menu (accessed using VoiceOver key + U and then using the cursor keys to cycle through the menu choices), you'll see that most of the elements are nicely listed so they can be accessed quickly.

However, we could do better here. The search form is a really important landmark that people will want to find, but it is not listed in the landmarks menu or treated like a notable landmark beyond the actual input being called out as a search input (<input type="search">).
To mark the form as a landmark, you can either wrap it with the <search> element or give it ARIA role="search". As a general rule, use HTML semantics where possible and only use ARIA where there is no HTML equivalent.
<header>
  <h1>Header</h1>
  <!-- Even is it's not mandatory, it's common practice to put the main navigation menu within the main header -->
  <nav>
    <ul>
      <li><a href="#">Home</a></li>
      <li><a href="#">Our team</a></li>
      <li><a href="#">Projects</a></li>
      <li><a href="#">Contact</a></li>
    </ul>
    <!-- A Search form is another common non-linear way to navigate through a website. -->
    <search>
      <form>
        <input
          type="search"
          name="q"
          placeholder="Search query"
          aria-label="Search through site content" />
        <input type="submit" value="Go!" />
      </form>
    </search>
  </nav>
</header>
<!-- Here is our page's main content -->
<main>
  <!-- It contains an article -->
  <article>
    <h2>Article heading</h2>
    <p>
      Lorem ipsum dolor sit amet, consectetur adipisicing elit. Donec a diam
      lectus. Set sit amet ipsum mauris. Maecenas congue ligula as quam viverra
      nec consectetur ant hendrerit. Donec et mollis dolor. Praesent et diam
      eget libero egestas mattis sit amet vitae augue. Nam tincidunt congue
      enim, ut porta lorem lacinia consectetur.
    </p>
    <h3>subsection</h3>
    <p>
      Donec ut librero sed accu vehicula ultricies a non tortor. Lorem ipsum
      dolor sit amet, consectetur adipisicing elit. Aenean ut gravida lorem. Ut
      turpis felis, pulvinar a semper sed, adipiscing id dolor.
    </p>
    <p>
      Pelientesque auctor nisi id magna consequat sagittis. Curabitur dapibus,
      enim sit amet elit pharetra tincidunt feugiat nist imperdiet. Ut convallis
      libero in urna ultrices accumsan. Donec sed odio eros.
    </p>
  </article>
  <!-- the aside content can also be nested within the main content -->
  <aside>
    <h2>Related</h2>
    <ul>
      <li><a href="#">Oh I do like to be beside the seaside</a></li>
      <li><a href="#">Oh I do like to be beside the sea</a></li>
      <li><a href="#">Although in the North of England</a></li>
      <li><a href="#">It never stops raining</a></li>
      <li><a href="#">Oh well...</a></li>
    </ul>
  </aside>
</main>
<!-- And here is our main footer that is used across all the pages of our website -->
<footer>
  <p>©Copyright 2050 by nobody. All rights reversed.</p>
</footer>
Most importantly, we have used semantic HTML that gives meaning and roles to the structure of the page without adding unnecessary role attributes to our HTML structure, which has a structure like this:
<header>
  <h1>…</h1>
  <nav>
    <ul>
      …
    </ul>
    <search>
      <form>
        <!-- search form -->
      </form>
    </search>
  </nav>
</header>
<main>
  <article>…</article>
  <aside>…</aside>
</main>
<footer>…</footer>
We've also given you a bonus feature in this example — the <input> element has been given the attribute aria-label, which gives it a descriptive label to be read out by a screen reader, even though we haven't included a <label> element. In cases like these, this is very useful — a search form like this one is a very common, easily recognized feature, and adding a visual label would spoil the page design.
<input
  type="search"
  name="q"
  placeholder="Search query"
  aria-label="Search through site content" />
Now if we use VoiceOver to look at this example, we get some improvements:
- The search form is called out as a separate item, both when browsing through the page, and in the Landmarks menu.
- The label text contained in the aria-labelattribute is read out when the form input is highlighted.
If you need to support older browsers such as IE8; it is worth including ARIA roles for that purpose. And if for some reason your site is built using just <div>s, you should definitely include the ARIA roles to provide these much needed semantics!
You'll see a lot more about these semantics and the power of ARIA properties/attributes below, especially in the Accessibility of non-semantic controls section. For now though, let's look at how ARIA can help with dynamic content updates.
Dynamic content updates
Content loaded into the DOM can be easily accessed using a screen reader, from textual content to alternative text attached to images. Traditional static websites with largely text content are therefore easy to make accessible for people with visual impairments.
The problem is that modern web apps are often not just static text — they often update parts of the page by fetching new content from the server (in this example we are using a static array of quotes) and updating the DOM. These are sometimes referred to as live regions.
Let's look at an example — a random quote generator:
<section>
  <h1>Random quote generator</h1>
  <button>Start giving me quotes</button>
  <blockquote>
    <p></p>
  </blockquote>
</section>
let quotes = [
  {
    quote:
      "Every child is an artist. The problem is how to remain an artist once he grows up.",
    author: "Pablo Picasso",
  },
  {
    quote:
      "You can never cross the ocean until you have the courage to lose sight of the shore.",
    author: "Christopher Columbus",
  },
  {
    quote:
      "I love deadlines. I love the whooshing noise they make as they go by.",
    author: "Douglas Adams",
  },
];
const quotePara = document.querySelector("section p");
const btn = document.querySelector("button");
btn.addEventListener("click", () => {
  function showQuote() {
    let random = Math.floor(Math.random() * quotes.length);
    quotePara.textContent = `${quotes[random].quote} -- ${quotes[random].author}`;
  }
  showQuote();
  btn.disabled = true;
  window.setInterval(showQuote, 5000);
});
This works OK, but it is not good for accessibility — the content update is not detected by screen readers, so their users would not know what is going on. This is a fairly trivial example, but just imagine if you were creating a complex UI with lots of constantly updating content, like a chat room, or a strategy game UI, or a live updating shopping cart display — it would be impossible to use the app in any effective way without some kind of way of alerting the user to the updates.
WAI-ARIA, fortunately, provides a useful mechanism to provide these alerts — the aria-live property. Applying this to an element causes screen readers to read out the content that is updated. How urgently the content is read out depends on the attribute value:
- off
- 
The default. Updates should not be announced. 
- polite
- 
Updates should be announced only if the user is idle. 
- assertive
- 
Updates should be announced to the user as soon as possible. 
Here we update the <blockquote> opening tag as follows:
<blockquote aria-live="assertive">…</blockquote>
This will cause a screen reader to read out the content as it is updated: try testing the updated the updated live version:
Note:
There are some other ARIA properties related to aria-live that are also worth knowing about:
- The aria-atomicproperty, when set totrue, tells screen readers to read out the entire element contents as one atomic unit, not just the bits that were updated. This is useful when only a section's contents are being updated, but you also want the heading to be read out each time something changes, to remind the user of its content.
- The aria-relevantproperty is useful for controlling what gets read out when a live region is updated. You can for example only get content additions or removals read out.
Enhancing keyboard accessibility
As discussed in a few other places in the module, one of the key strengths of HTML with respect to accessibility is the built-in keyboard accessibility of features such as buttons, form controls, and links. Generally, you can use the tab key to move between controls, the Enter/Return key to select or activate controls, and occasionally other controls as needed (for example the up and down cursor to move between options in a <select> box).
However, sometimes you will end up having to write code that either uses non-semantic elements as buttons (or other types of control), or uses focusable controls for not quite the right purpose. You might be trying to fix some bad code you've inherited, or you might be building some kind of complex widget that requires it.
In terms of making non-focusable code focusable, WAI-ARIA extends the tabindex attribute with some new values:
- tabindex="0"— as indicated above, this value allows elements that are not normally tabbable to become tabbable. This is the most useful value of- tabindex.
- tabindex="-1"— this allows not normally tabbable elements to receive focus programmatically, e.g., via JavaScript, or as the target of links.
We discussed this in more detail and showed a typical implementation back in our HTML accessibility article — see Building keyboard accessibility back in.
Accessibility of non-semantic controls
This follows on from the previous section — when a series of nested <div>s along with CSS/JavaScript is used to create a complex UI-feature, or a native control is greatly enhanced/changed via JavaScript, not only can keyboard accessibility suffer, but screen reader users will find it difficult to work out what the feature does if there are no semantics or other clues. In such situations, ARIA can help to provide those missing semantics.
Form validation and error alerts
First of all, let's revisit the form example we first looked at in our CSS and JavaScript accessibility article (read Keeping it unobtrusive for a full recap). At the end of this section, we showed that we have included some ARIA attributes on the error message box that displays any validation errors when you try to submit the form:
<div class="errors" role="alert" aria-relevant="all">
  <ul></ul>
</div>
- role="alert"automatically turns the element it is applied to into a live region, so changes to it are read out; it also semantically identifies it as an alert message (important time/context-sensitive information), and represents a better, more accessible way of delivering an alert to a user (modal dialogs like- alert()calls have a number of accessibility problems; see Popup Windows by WebAIM).
- An aria-relevantvalue ofallinstructs the screen reader to read out the contents of the error list when any changes are made to it — i.e., when errors are added or removed. This is useful because the user will want to know what errors are left, not just what has been added or removed from the list.
We could go further with our ARIA usage, and provide some more validation help. How about indicating whether fields are required in the first place, and what range the age should be?
- 
At this point, take a copy of our form-validation.htmlandvalidation.jsfiles, and save them in a local directory.
- 
Open them both in a text editor and have a look at how the code works. 
- 
First of all, add a paragraph just above the opening <form>tag, like the one below, and mark both the form<label>s with an asterisk. This is normally how we mark required fields for sighted users.html<p>Fields marked with an asterisk (*) are required.</p>
- 
This makes visual sense, but it isn't as easy to understand for screen reader users. Fortunately, WAI-ARIA provides the aria-requiredattribute to give screen readers hints that they should tell users that form inputs need to be filled in. Update the<input>elements like so:html<input type="text" name="name" id="name" aria-required="true" /> <input type="number" name="age" id="age" aria-required="true" />
- 
If you save the example now and test it with a screen reader, you should hear something like "Enter your name star, required, edit text". 
- 
It might also be useful if we give screen reader users and sighted users an idea of what the age value should be. This is often presented as a tooltip or placeholder inside the form field. WAI-ARIA does include aria-valueminandaria-valuemaxproperties to specify min and max values, and screen readers support the nativeminandmaxattributes. Another well-supported feature is the HTMLplaceholderattribute, which can contain a message that is shown in the input when no value is entered and is read out by a few screen readers. Update your number input like this:html<label for="age">Your age:</label> <input type="number" name="age" id="age" placeholder="Enter 1 to 150" required aria-required="true" />
Always include a <label> for every input. While some screen readers announce the placeholder text, most do not. Acceptable substitutions for providing form controls with an accessible name include aria-label and aria-labelledby. But the <label> element with a for attribute is the preferred method as it provides usability for all users, including mouse users.
Note:
You can see the finished example live at form-validation-updated.html.
WAI-ARIA also enables some advanced form labelling techniques, beyond the classic <label> element. We already talked about using the aria-label property to provide a label where we don't want the label to be visible to sighted users (see the Signposts/Landmarks section, above). Some other labeling techniques use other properties such as aria-labelledby if you want to designate a non-<label> element as a label or label multiple form inputs with the same label, and aria-describedby, if you want to associate other information with a form input and have it read out as well. See WebAIM's Advanced Form Labeling article for more details.
There are many other useful properties and states too, for indicating the status of form elements. For example, aria-disabled="true" can be used to indicate that a form field is disabled. Many browsers will skip past disabled form fields which leads to them not being read out by screen readers. In some cases, a disabled element will be perceived, so it is a good idea to include this attribute to let the screen reader know that a disabled form control is in fact disabled.
If the disabled state of an input is likely to change, then it is also a good idea to indicate when it happens, and what the result is. For example, in our form-validation-checkbox-disabled.html demo, there is a checkbox that when checked, enables another form input to allow further information to be entered. We've set up a hidden live region:
<p class="hidden-alert" aria-live="assertive"></p>
which is hidden from view using absolute positioning. When this is checked/unchecked, we update the text inside the hidden live region to tell screen reader users what the result of checking this checkbox is, as well as updating the aria-disabled state, and some visual indicators too:
function toggleMusician(bool) {
  const instrument = formItems[formItems.length - 1];
  if (bool) {
    instrument.input.disabled = false;
    instrument.label.style.color = "black";
    instrument.input.setAttribute("aria-disabled", "false");
    hiddenAlert.textContent =
      "Instruments played field now enabled; use it to tell us what you play.";
  } else {
    instrument.input.disabled = true;
    instrument.label.style.color = "#999999";
    instrument.input.setAttribute("aria-disabled", "true");
    instrument.input.removeAttribute("aria-label");
    hiddenAlert.textContent = "Instruments played field now disabled.";
  }
}
Describing non-semantic buttons as buttons
A few times in this course already, we've mentioned the native accessibility of (and the accessibility issues behind using other elements to fake) buttons, links, or form elements (see Use semantic UI controls where possible in the HTML accessibility article, and Enhancing keyboard accessibility, above). Basically, you can add keyboard accessibility back in without too much trouble in many cases, using tabindex and a bit of JavaScript.
But what about screen readers? They still won't see the elements as buttons. If we test our fake-div-buttons.html example in a screen reader, our fake buttons will be reported using phrases like "Click me!, group", which is obviously confusing.
We can fix this using a WAI-ARIA role. Make a local copy of fake-div-buttons.html, and add role="button" to each button <div>, for example:
<div data-message="This is from the first button" tabindex="0" role="button">
  Click me!
</div>
Now when you try this using a screen reader, you'll have buttons be reported using phrases like "Click me!, button". While this is much better, you still have to add in all the native button features users expect, like handling enter and click events, as explained in the button role documentation.
Guiding users through complex widgets
There are a whole host of other roles that can identify non-semantic element structures as common UI features that go beyond what's available in standard HTML, for example combobox, slider, tabpanel, tree. You can see several useful examples in the Deque university code library to give you an idea of how such controls can be made accessible.
You can also find several live examples in our WAI-ARIA roles documentation. See for example, our ARIA: tab role example, which explains how to implement an accessible tabbed interface.
Summary
This article has by no means covered all that's available in WAI-ARIA, but it should have given you enough information to understand how to use it, and know some of the most common patterns you will encounter that require it.
In the next article, we'll give you some tests that you can use to check how well you've understood and retained all this information.
See also
- Aria states and properties: All aria-*attributes
- WAI-ARIA roles: Categories of ARIA roles and the roles covered on MDN
- ARIA in HTML on W3C: A specification that defines, for each HTML feature, the accessibility (ARIA) semantics implicitly applied on it by the browser and the WAI-ARIA features you may set on it if extra semantics are required
- Deque university code library: A library of really useful and practical examples showing complex UI controls made accessible using WAI-ARIA features
- WAI-ARIA authoring practices on W3C: A very detailed design pattern from the W3C, explaining how to implement different types of complex UI control whilst making them accessible using WAI-ARIA features