Author: Kevin Babbitt, Melanie Richards
Last updated: 2021-01-07
This document is intended as a starting point for engaging the community and standards bodies in developing collaborative solutions fit for standardization. As the solutions to problems described in this document progress along the standards-track, we will retain this document as an archive and use this section to keep the community up-to-date with the most current standards venue and content location of future work and discussions.
- This document status: ARCHIVED
- Current venue: W3C Web Incubator Community Group
- Current version: https://github.com/WICG/accessible-loading-and-searching-of-content/blob/master/explainer.md
Virtualized content allows an author to efficiently represent a large body of information at once. In this practice, only a fraction of the information is actually present in the markup - typically the portion that is currently visible, plus a small amount preceding or following in order to allow for scrolling. As the user scrolls the web page, JavaScript code will load more content and insert it into the markup at the appropriate location. This process is referred to as realizing the content.
Virtualized content presents challenges for users of assistive technologies (ATs). Generally, an AT's view of the document is limited to what is physically present in the markup. Without some indication that further content exists, the AT cannot accurately describe and navigate the web page. For example, some ATs allow the user to skim the headings in a document by jumping from one to the next. After reading the last realized heading, the AT has no way of knowing that another heading might exist in virtualized content, nor does it have a way of discovering that there is even virtualized content to begin with. The AT, and the user, thus incorrectly conclude that the last heading in the document has been reached.
- Enable ATs to recognize and account for virtualized content when navigating a web page.
- Protect the privacy of AT users by avoiding mechanisms that would allow for fingerprinting.
A new attribute, aria-virtualcontent
, indicates whether an element contains virtualized content. The default value is none
.
A virtual content container is any element that has a non-default value for aria-virtualcontent
.
A virtual content edge is an edge of a virtual content container where the webpage can realize content.
The aria-virtualcontent
attribute can have one of several keyword values, or multiple values separated by spaces, establishing which edges of the container are virtual content edges: block-end
, block-start
, inline-end
, inline-start
.
For potential future expansions of this API, such as hint(s) about the nature of the virtualized content, please refer to Potential additions to proposed API.
Marking an element as a virtual content container establishes a contract between the web page and ATs. Specifically:
- If the virtual content container is also a scroll container, the web page MUST begin steps to realize content no later than when the virtual content container is scrolled to a limit where a virtual content edge exists.
- If the virtual content container is not a scroll container, the web page MUST begin steps to realize content no later than when a virtual content edge enters the scrollport of its nearest ancestor scroll container.
The following simplified example shows one potential usage pattern for aria-virtualcontent
.
<html>
<head>
<script src="virtualcontent.js"></script>
</head>
<body>
<main id="main" aria-virtualcontent="block-end" aria-busy="false">
<h1>Section 1</h1>
...
<h1>Section 2</h1>
<h2>Section 2.1</h2>
...
<!-- etc... --->
<div id="loading_spinner" role="presentation" style="visibility:hidden">
<img src="loading.gif" role="presentation">
</div>
</main>
<div id="announcer" aria-live="assertive" style="width: 0; height: 0; overflow: hidden"></div>
</body>
</html>
virtualcontent.js:
// Issues a request to the server for the next section of content.
function begin_load_next_section() {
// Issue request to server.
// ...
}
// Callback triggered when the server responds with next section of content.
function end_load_next_section(response) {
// Signal to the AT that we're loading more content.
document.getElementById("main").setAttribute("aria-busy", "true");
// Insert response payload into the DOM.
document.getElementById("main").appendChild(...);
// Signal to the AT that we're done loading.
document.getElementById("main").setAttribute("aria-busy", "false");
// Signal to the user that we're done loading.
document.getElementById("loading_spinner").style.visibility = "hidden";
document.getElementById("announcer").textContent = "Next section loaded.";
// Update the state of the virtual content container.
if (response.isAtEndOfDocument) {
document.getElementById("main").removeAttribute("aria-virtualcontent");
}
}
// Listens for scroll events and responds accordingly.
function on_scroll_changed() {
// Check whether the user has scrolled to the bottom of the page.
let target = document.documentElement;
let top = (target && target.scrollTop) || document.body.scrollTop;
if (top + target.clientHeight >= target.scrollHeight) {
// Issue request to the server.
begin_load_next_section();
// Signal to the user that we're loading more content.
document.getElementById("loading_spinner").style.visibility = "visible";
document.getElementById("announcer").textContent = "Loading additional content.";
}
}
// Initialize scroll handler.
window.addEventListener("scroll", on_scroll_changed);
- The web page is in a steady state.
- The
main
element is partially visible in the viewport. This element's bottom edge is below the bottom edge of the viewport. Because it hasaria-virtualcontent
set, it is a virtual content container. - An AT has its reading cursor on the last heading element inside the virtual content container.
- The user directs the AT to navigate to the next heading in the document.
- Using the platform accessibility API, the AT searches the web page for a subsequent heading. It finds none.
- The AT searches the parent chain of the last heading element (i.e. the current location of its reading cursor). It finds the
main
element and sees that it is a virtual content container. - The AT reacts to the presence of a virtual content container by looking for a scrollable element, starting with the virtual content container itself and searching the parent chain from there. It finds the scroller for the document itself and asks the user agent to scroll to the bottom of the page. (Privacy note: APIs already exist on some operating systems that allow an AT to do this without fingerprinting the user.)
- The user agent scrolls as requested by the AT and generates a scroll event.
- The
on_scroll_changed()
function is called in reaction to the scroll event. Script code issues a request to a backing server for additional content, displays a loading spinner, and announces that additional content is being loaded using an ARIA live region. - The backing server sends back a payload with additional content. Upon receiving the payload, the
end_load_next_section()
function is called. Script code inserts the new content into the DOM at the bottom of the virtual content container, hides the loading spinner, and announces to the user that additional content is available. - The user agent notifies the platform accessibility API that the web page content has changed.
- The user hears the announcement that new content has been loaded. He/she once again directs the AT to navigate to the next heading.
- Using the platform accessibility API, the AT searches the web page for a subsequent heading.
- The AT finds a heading in the newly loaded content, moves its reading cursor to that heading, and reads the heading to the user.
- The user continues navigating through document headings in the same fashion.
- Eventually, the backing server sends a content payload which also contains a signal that the end of the document has been reached. Script code removes the
aria-virtualcontent
attribute from themain
element to indicate there is no more virtualized content. - The next time the user attempts to navigate to the next heading, the AT searches for a virtual content container, finds none, and announces there are no further headings in the document.
This example illustrates one use of aria-virtualcontent
for virtualized content in the inline direction. The document in this scenario is a table of bug reports with the following columns: report ID, report date, status, assigned to, title, fix date, fixed by.
In a fully-realized table, a typical AT would walk the table in row-major order, stopping and reading out each cell in turn. In this example, the content author has incorporated script that keeps only a subset of columns realized depending on viewport width, and the viewport width is such that the script will keep no more than five columns realized at a time. The AT achieves the same reading order as for a fully-realized table by checking for virtualized content in the inline direction whenever it's ready to advance from the last realized cell in a row.
<html>
<body>
<table aria-virtualcontent="block-end inline-end" aria-colcount="7" aria-busy="false">
<thead>
<tr>
<th>ID</th>
<th>Status</th>
<th>Assigned To</th>
<th>Report Date</th>
<th>Title</th>
<!-- <th>Fix Date</th> - Virtualized -->
<!-- <th>Fixed By</th> - Virtualized -->
</tr>
</thead>
<tbody>
<tr>
<td>338</td>
<td>Closed</td>
<td>Alice</td>
<td>April 21</td>
<td>Widget freezes up when the network is slow</td>
<!-- <td>Fix Date</td> - Virtualized -->
<!-- <td>Fixed By</td> - Virtualized -->
</tr>
<tr>
<td>342</td>
<td>Fixed</td>
<td>Bob</td>
<td>April 22</td>
<td>Crash when shift-double-clicking the widget</td>
<!-- <td>Fix Date</td> - Virtualized -->
<!-- <td>Fixed By</td> - Virtualized -->
</tr>
</tbody>
</table>
</body>
</html>
Backing script performing similar functions as Example 1's virtualcontent.js is assumed to be present.
An AT navigating the above content by table cells might result in the following flow:
- The AT stops on and reads out each of the five realized cells in the header row.
- Upon reaching the end of the row, the AT discovers that the table has virtualized content in the inline-end direction and scrolls the document in that direction.
- Backing script realizes headers and data for the "Fix date" and "Fixed by" columns. It also virtualizes headers and data for the "ID" and "Status" columns. There are no longer any virtualized columns in the inline-end direction, but there are now virtualized columns in the inline-start direction. Accordingly, in the table's
aria-virtualcontent
attribute, script replaces theinline-end
token withinline-start
. - The AT stops on and reads out the two newly realized cells in the header row.
- Upon reaching the end of the row, the AT discovers that the table no longer contains virtualized content in the inline-end direction. The next realized cell is the "Assigned To" cell in the row for bug 338. However, the table contains virtualized content in the inline-start direction, so the AT recognizes that the next realized cell is not the next cell in the overall table. The AT notes to itself that it is moving into the row for bug 338 and scrolls the document in the inline-start direction.
- Backing script realizes headers and data for the "ID" and "Status" columns. It also virtualizes headers and data for the "Fix date" and "Fixed by" columns. There are no longer any virtualized columns in the inline-start direction, but there are now virtualized columns in the inline-end direction. Accordingly, in the table's
aria-virtualcontent
attribute, script replaces theinline-start
token withinline-end
. - The AT moves to the first realized cell in the row for bug 338. In turn, it stops on and reads out each of the realized cells in that row.
- After reading the last realized cell in the row for bug 338, the AT checks for virtualized content in the inline-end direction and finds there is some. It scrolls the document in the inline-end direction.
- Backing script realizes content in the inline-end direction, virtualizes content in the inline-start direction, and updates the table's
aria-virtualcontent
attribute, the same as in step 3. - The AT stops on and reads out the two newly realized cells in the row for bug 338.
- Upon reaching the end of the row, the AT discovers that the table no longer contains virtualized content in the inline-end direction, then moves on to the next row (the row for bug 342) and checks for virtualized content in the inline-start direction, the same as in step 5.
- Backing script realizes content in the inline-start direction, virtualizes content in the inline-end direction, and updates the table's
aria-virtualcontent
attribute, the same as in step 6. - The AT stops on and reads out each cell in the row for bug 342, scrolling as necessary to realize additional cells. The flow is the same as for the previous row, as outlined in steps 7-10.
- After reading the last cell in the row for bug 342, the AT discovers that the table no longer contains virtualized content in the inline-end direction. It looks for another row in the table and finds there are no more realized rows.
- However, the AT also discovers that the table has virtualized content in the block-end direction. The AT issues two scroll requests: first in the inline-start direction to return to the first columns in the table, then in the block-end direction to load additional rows.
- Backing script realizes the next few rows in the table, and the AT continues reading.
The proposed API aims for flexibility in assistive technologies’ user experiences. If deemed useful by the community, the API could potentially be extended to provide more precise hints.
The aria-virtualcontent
attribute or a related attribute (e.g. aria-virtualcontenttypes
)
could provide a hint to assistive technologies as to important types of content that are currently virtualized.
Such a hint could take space-separated keywords as its value (e.g. headings
, tables
),
and a default, implicit value of any
.
Assistive technologies could use this hint when implementing various navigational modes,
such as "move by heading".
This token list would necessarily need to be scoped to a limited subset of keywords,
to avoid introducing extensive microformats.
WAI ARIA 1.1 defines the feed
role, which
implements one solution for allowing ATs to interact with virtualized content, in the context of a scrollable list of articles.
However, feed
has limitations:
- Because it is implemented as a role, it cannot coexist with other roles. This limits its applicability - for example, one might
imagine virtualizing portions of a spreadsheet whose role is best defined as
table
orgrid
. - There is no explicit indication that virtualized content exists; in other words it does not allow the AT to distinguish between the end of realized content and the actual end of content. This can result in the AT giving incorrect cues to the user as described in the introduction.