How can I make screen readers respond to showing and hiding content in a dynamic web application?
Asked Answered
D

1

6

I would like to create an accessible web page which contains a number of components that can be hidden and shown as the user interacts with the page. When a component is shown I expect a screen reader (in this case NVDA) to read the contents of the component.

As an example:

<html>
<body>
	<div id="comp1" style="display:none;" role="region" tabindex="-1" aria-expanded="false">
		<div>This is component 1</div>	
		<button type="button" onclick="ShowComponent(comp2)">Show 2</button>	
	</div>
	<div id="comp2" style="display:none;" role="region" tabindex="-1" aria-expanded="false">
		<div>This is component 2</div>
		<button type="button" onclick="ShowComponent(comp1)">Show 1</button>
	</div>

	<script type="text/javascript">
		function HideAll() {		
			// hide both components
			var comp1 = document.getElementById("comp1");
			comp1.style.display = "none";
			comp1.setAttribute("aria-expanded", "false");
			
			var comp2 = document.getElementById("comp2");
			comp2.style.display = "none";
			comp2.setAttribute("aria-expanded", "false");
		}
		
		function ShowComponent(comp) {
			HideAll();
			
			// show this component
			comp.style.display = "block";
			comp.setAttribute("aria-expanded", "true");
			comp.focus();			
		}	
		
		ShowComponent(document.getElementById("comp1"));
	</script>
</body>
</html>

In the above example clicking a button within component 1 will hide component 1 and show component 2.

Clicking a button within component 2 will hide component 2 and show component 1.

However NVDA does not seem to read the contents of the components when toggling between the components. Is there a way that this can be achieved?

Please see this Gist on GitHub to check using NVDA

Declarative answered 23/11, 2016 at 13:53 Comment(0)
A
5

Stuff those two blocks into an ARIA live region.

There are some live region properties you need to know to make it work. If you want them to be announced as soon as they change, no matter what the user is doing, then it will be assertive. If you want it to wait until the user finishes an interaction, then it will be polite. There are corresponding live region roles as well, which I show below.

I do not think you need the other ARIA bits nor tabindex in your example code, but I do not know the scope of the page. Here is a different example instead:

<div role="alert" aria-live="assertive">
Anything in here will be announced as soon as it changes.
</div>

<div role="status" aria-live="polite">
Anything in here will be announced when it changes but only after the user has stopped interacting with the page..
</div>

Further, the aria-atomic property will tell the screen reader whether it should announce the entire thing (which is good for an alert message) or just the parts that changed (which might be a better fit for a timer).

Avoid having more than one ARIA live region on a page.

This will also address screen readers besides NVDA.

Here is an example of an offline alert. Here is a handy slideshare that goes into more detail. There is a lot more out there now that you know what to search.

Aylmar answered 23/11, 2016 at 14:23 Comment(2)
Thanks for your response. Adding a live region around my components fixed my problem and NVDA now reacts as expected. I had previously experimented with putting the aria-live attribute on the components themselves, but without success. It appears that the main problem I was having was with the scope of my live region!Declarative
Yep, that is a common issue. Knowing what to wrap is huge, and then knowing how aggressive (assertive v. polite) and how detailed (atomic) can be tricky until you test it with users. Good luck!Aylmar

© 2022 - 2024 — McMap. All rights reserved.