Accessibility Object Model

Specification Editors:
Alexander Surkov, Mozilla
Alice Boxhall, Google
Dominic Mazzoni, Google
James Craig, Apple

Presentation Check

Great! The Accessibility Object Model is enabled in this browser.

Demos should work fine.

Accessibility Object Model support is not enabled. Interactive features of these slides won't work.

To enable it in Chrome, use this command-line switch:

        --enable-blink-features=AccessibilityObjectModel
      

To enable it in Safari Technology Preview, select
Develop > Experimental Features > Accessibility Object Model.

Tip: Arrow keys move between slides, but not when something else has focus.

Page Up / Down moves between slides, regardless of focus.

Overview

Background

Assistive technology (AT) augments or replaces the existing UI for an application. Examples include:

Image showing 4 layers of 2-way communication - from the application, to the accessibility tree, to assistive technology, to the user.

Accessibility APIs

Each operating system has its own accessibility APIs for native applications.

Besides assistive technology, these APIs are also often used for automation, for example for password managers or for single-sign-on.

Accessibility Tree

Native applications expose an accessibility tree via accessibility APIs. Each accessibility node has properties such as a role, name, state, and actions.

Visualization of an accessibility tree. The root has role main, the child has role form, then it has three children, a radio button, a slider, and a button, each with role, name, state, and actions pictured.

Accessibility on the web

Web browsers automatically map the DOM into the accessibility tree.

Image showing how the web browser simultaneously paints the visual UI and also updates the accessibility tree.

Web accessibility APIs

The web has rich support for making applications accessible, but only via a declarative API.

There's always a one-to-one correspondence between a DOM node and a node in the accessibility tree.

Image showing how a div with role=checkbox might simultaneously be drawn as a custom checkbox, while mapping onto a native chcekbox in the accessibility API.

Other platform accessibility APIs

Typically, most platforms allow developers to use a programmatic API to create virtual accessibility trees to implement accessibility for custom views.

The web platform requires everything to do with accessibility to be specified declaratively, even as many parts of the web platform are moving to more dynamic, imperative style APIs.

Motivating Use Cases

Use Case 1: Setting non-reflected (“default”) accessibility properties for Web Components

Today, a library author creating a custom element is forced to "sprout" ARIA attributes to express semantics which are implicit for native elements.

    
    

    
    
  

Unlike a native element, a custom element's semantics may be completely overwritten or removed by the author using the custom element, and it is impossible to distinguish between the semantics provided by the custom element and semantics set by the page author.

This capability need not, but may be limited to Web Components.

Use Case 2: Setting relationships without IDREFs

To express an accessible relationship between two elements in the DOM, ARIA attributes let one element refer to the ID of another element.

aria-labelledby indicates one element labels another:

    
First name:

aria-activedescendant indicates a descendant that's focused in a composite control like a list box.

    
Item 1
Item 2
Item 3

Use Case 2: Setting relationships without IDREFs (2)

It can be cumbersome and error-prone to maintain unique IDs on all elements that have a relationship with some other element.

But if you're using Shadow DOM, it's impossible to establish a relationship between two elements that aren't part of the same tree scope.

For example, it's impossible to make an accessible combobox using aria-activedescendant if the text field is in a different tree scope from the listbox options:

      
          #shadow-root (open)
          |  
          |  
          |  
          
            Option 1
            Option 2
            Option 3
         
        
  

Use Case 3: Capturing input events from AT

ARIA only helps the web author communicate in one direction: from the web app to the assistive technology.

However, AT can also command and control applications, for example issuing commands to activate, focus, or scroll. While native HTML elements already support these commands, web authors can't override this behavior for custom elements.

Examples: AT wants to select an item in a list box, increment a slider, or dismiss a dialog.

Another example: AT user performs a gesture to explicitly scroll to the left on a map widget.

Use Case 4: Making custom-drawn UI accessible.

To make a canvas-based UI (for example, a UI drawn using WebGL) accessible requires hacks like non-rendered fallback DOM content doing double duty as the accessible version of the painted UI.

More generally, any UI which has a visual semantic structure (such as an image with complex content) which doesn't match up to its DOM structure is difficult to make accessible.

Use Case 5: Introspection

There's no way to programmatically detect if an accessibility feature is supported by the browser or if you're using it correctly.

    
    
    
    

Accessibility Object Model

AOM: Brief History

Microsoft and Mozilla independently proposed a JavaScript accessibility API. Apple and Google had similar ideas.

AOM is a collaboration between Apple, Google, and Mozilla, part of the Web Incubator Community Group. It takes the best ideas from previous proposals and organizes them into four phases, each with smaller scope, each solving an increasing number of real-world use-cases.

Both Safari and Google Chrome have implemented some of the spec experimentally.

Warning!

All syntax shown on these slides is subject to change.

We are committed to solving these problems, but the solution may not be exactly what you see here.

Phases 1-3 overview

Phases 1 through three, taken together, should bring the capabilities of accessibility on the web closer to what is possible on native platorms.

Phase 1: Accessible Properties

Basically, reflect ARIA into JavaScript.

Every ARIA attribute, such as role and aria-checked here:

    
Receive promotional offers

...can be set directly from JavaScript:

    el.role = "checkbox";
    el.ariaChecked = true;
  

Phase 1: Accessible Properties (2)

These properties will also be available on the ElementInternals interface. This will allow Custom Elements to express their intrinsic semantics:

          class CustomCheckbox extends HTMLElement {
            constructor() {
              super();
              this._internals = this.attachInternals();
              this._internals.role = "checkbox";  // Set the default semantics.
            }
          
            // When the custom "checked" attribute changes, keep the accessible checked state in sync.
            attributeChangedCallback(name, oldValue, newValue) {
              switch(name) {
              case "checked":
                this._internals.ariaChecked = (newValue !== null);
              }
            }
          }
          // Authors may use ARIA as usual: 
          customElements.define("custom-checkbox", CustomCheckbox);
      

Phase 1: Accessible Properties (3)

Instead of using an IDREF:

    el.setAttribute('aria-activedescendant', 'child' + childIndex);
  

...just reference another element directly:

    el.ariaActiveDescendant = child;
  

Phase 2: Background on events from AT

Problem: if you create an accessible element with role=slider, then interact with that element via a screen reader on a mobile device, you will hear instructions to increment or decrement the slider.

However, there is no way for a web developer to implement the slider such that following those instructions will have the desired effect.

Phase 2: Background on events from AT

As a reminder, AT users are often not generating keyboard, mouse, and touch events.

They issue high-level commands (click, focus, select, dismiss, etc.) directly to their AT, and the AT tells the app to take that action via accessibility APIs.

Phase 2: Accessible Actions / Input Events

To preserve the privacy of assistive technology users, events from assistive technology will cause a synthesised DOM event to be triggered where appropriate:

    slider.addEventListener("keydown", function(event) {
      // Keydown event may be generated via keyboard or via assistive technology gesture
      console.log("Got keydown event");
      switch (event.code) {
        case "ArrowUp":
          customSlider.value += 1;
          return;
        case "ArrowDown":
          customSlider.value -= 1;
          return;
      }
    });
    

Phase 2: Accessible Actions / Input Events

New input event types capturing common patterns will also be added. Where supported, these new event types will be triggered any user interactions which triggers the same events as those which would be fired as a fallback.

For example, if an "increment" gesture from AT triggers a keydown/keyup sequence for the right arrow key, then if a user presses the right arrow key while focused on an applicable element, an increment event will also be fired.

    // "increment" or "decrement" may be triggered by an AT gesture, by a keyboard event or 
    // by a specific interaction such as clicking a button

    slider.addEventListener("increment", function(event) {
      customSlider.value += 1;
    });

    slider.addEventListener("decrement", function(event) {
      customSlider.value -= 1;
    });

Phase 3: Virtual Nodes

You can construct a "virtual" AccessibleNode that isn't associated with any DOM element.

    let virtualNode = new AccessibleNode();
    virtualNode.role = "button";
    virtualNode.label = "Play Game";
  

Insert an AccessibleNode as the child of a DOM element or another AccessibleNode:

    // This replaces all DOM children of document.body for the purposes
    // of accessibility tree computation.
    document.body.attachAccessibleRoot();
    document.body.accessibleRoot.appendChild(virtualNode);
  

Restrictions on virtual nodes

You can't use these methods to rearrange the accessibility tree corresponding to the DOM.

You can only build a "virtual" accessibility tree that "hangs off" of one DOM node.

Bounding boxes

To set the bounding box of a virtual node, provide its left, top, width, and height in pixels, relative to its offsetParent, which can be any ancestor AccessibleNode.

    virtualNode.offsetLeft = 30;
    virtualNode.offsetTop = 20;
    virtualNode.offsetWidth = 400;
    virtualNode.offsetHeight = 300;
    virtualNode.offsetParent = document.body.accessibleRoot.firstChild;
  

Relative coordinates are important for things like a canvas-based UI, because the canvas element might have CSS transforms applied to it.

Virtual AccessibleNode demo

Tic-tac-toe

Phase 4: Introspection

Every AccessibleNode has a corresponding ComputedAccessibleNode.

Accessing it is asynchronous (via a Promise), and it gives you full access to that node, and the accessibility tree.

    let computed = await getComputedAccessibleNode(myListItem);
    computed.role;             // "listitem"
    computed.name;             // "Picard"
    computed.parent.role;      // "list"
    computed.children.length;  // 0
  

Phase 4: Feature detection

The ComputedAccessibleNode enables you to determine if a role, or any other accessibility feature, is supported.

    let button = document.createElement("button");
    button.role = "hammer";
    let computed = await getComputedAccessibleNode(button);
    computed.role;             // "button" (not "hammer")
  

Phase 4: Walking the tree

Walk the accessibility tree for testing or debugging.

    function printAccessibilityTree(computed, indent) {
      print(spaces(indent) + 'Role: ' + computed.role + ', Name: ' + computed.name);
      for (let i = 0; i < computed.children.length; i++) {
        printAccessibilityTree(computed.children[i], indent + 1);
      }
    }

    let computed = await getComputedAccessibleNode(document.body);
    printAccessibilityTree(computed, 0);
  

Phase 4: Writing in-browser AT

Write a screen reader or other AT

AOM Screen Reader

For more information:
github.com/WICG/aom

Questions?