Testing Accessibility with Screen Readers

Shruti Kapoor
AuthorShruti Kapoor

Accessibility is a key component of building web applications. Prioritizing accessibility benefits everyone—both people with disabilities and able-bodied individuals—by creating a more efficient, usable and inclusive experience for all.

If you're new to accessibility, I recommend starting with my previous article in this series - Testing Accessibility with the Keyboard, where I covered how to build keyboard-accessible web applications. Incorporating keyboard accessibility improves web accessibility significantly. However, it is only the first step.

This guide draws on my experience working on the accessibility team at Slack, to develop web applications that meet accessibility standards. Here, I’ll outline how screen readers work, share practical testing workflows, and provide tips for making your applications more accessible.

How do users use screen readers?

Screen readers are vital assistive technologies used primarily by people with low vision or complete blindness to navigate websites. Screen readers offer a range of navigation modes, such as quickly jumping between headings, links, or landmarks which significantly improves the browsing experience for users. They enable users to:

  • Navigate content sequentially or by specific elements (e.g., headings, links).
  • Understand the role, name, and state of interactive elements.
  • Use keyboard shortcuts for efficient browsing.

Here’s a video of a visually impaired person using a screen reader to demonstrate how users use screen readers.

Some screen readers that are commonly used are -

  1. Apple Voiceover
  2. NVDA
  3. JAWS
  4. Talkback (Android)
  5. ChromeVox

Testing with only one screen reader is not sufficient for accessibility testing. Each has its own quirks, so it’s important to test with multiple screen readers.

Screen reader modes

In order to make navigation and reading easier, screen readers provide two modes of navigation - browse mode and focus/application mode.

  1. Browse mode:
    • Used for navigating and reading the page
    • Supports navigating by headings, links, and regions and provides keyboard shortcuts to facilitate navigation
    • Ideal for exploring pages with a lot of static content
  2. Focus/Application mode:
    • Engages directly with interactive elements like forms or modals
    • Disables shortcuts for moving between elements
    • Helpful when interacting with an application with lots of interactive and dynamic elements.

Technical overview of how screen readers work

Screen readers navigate the UI from top to bottom, providing users with enough information to understand and interact with the content on the page by leveraging the accessibility tree.

What is the accessibility tree?

The accessibility tree is a simplified, structured representation of the Document Object Model (DOM) that assistive technologies, like screen readers, use to interpret the UI. While it originates from the DOM, it is not a 1:1 copy of the DOM tree. Instead, the browser refines the DOM tree into a format that omits unnecessary or purely decorative information, focusing on content that is meaningful and interactive.

For instance, visual elements like decorative images or icons with no functional purpose are excluded from the accessibility tree. Therefore, the accessibility tree is similar to the DOM tree but with less information. The goal is to provide just enough context about each node without excessive verbosity, making navigation intuitive and efficient.

Here’s a representation of what the accessibility tree looks like -

a11y tree Source: https://web.dev/articles/the-accessibility-tree

Assistive technologies, including screen readers, switch devices, and braille displays, depend on the accessibility tree—not the DOM—to understand and present the content of a web page. A well-structured accessibility tree ensures that users with disabilities can effectively interact with applications.

To view the accessibility tree in browser developer tools:

  • Chrome:
    • Open DevTools → "Elements" tab → "Accessibility" pane

    • Check “Enable full-page accessibility tree”

    • Click on the man icon

      man icon

  • Firefox:
    • Right click on a page → Inspect Accessibility Properties

a11y tree The accessibility tree when viewed on Google.com in Google Chrome

How Screen Readers use the accessibility tree

The browser generates the accessibility tree by combining the DOM, CSS, and ARIA attributes. It then creates platform-specific accessibility APIs (e.g., macOS Accessibility API, Android Accessibility Framework), etc. , which assistive technologies use to retrieve and convey content.

In order to convey complete information to the user, a screen reader needs the following information from the accessibility tree -

  • Role: Defines the element's type (e.g., “button”, “heading”, “link”).
  • Name: Specifies the element's accessible label or text content (e.g., button text or aria-label).
  • State: Shows the element's current condition (e.g., “checked", “expanded", “disabled”).
  • Properties: Provides additional attributes, such as whether an element is required or hidden.
  • Hierarchy: Shows parent-child relationships to help screen readers understand content structure and flow.

For example, a button such as:


<button>Submit</button>

would appear in the accessibility tree as:

  • Role: Button
  • Name: Submit
  • State: Default (not pressed)

Why is there no aria-label and role added to button here? Since <button> is a semantic element, its role and label are automatically derived, making additional attributes like aria-label redundant.

A screen reader announces a DOM element in the following format -


[text label], [element type], [state]

Therefore, this button would be announced as -

“Submit, button”

screenshot a11y On the Google homepage, the “Google apps” button in its collapsed state is announced as: “Google apps, collapsed, button, group.”

Testing web applications with screen readers

To test with screen reader, you can first enable the screen reader on your computer and test your application by navigating via the keyboard and listening to the announcements of the screen reader.

Enabling VoiceOver

  1. Go to SettingsAccessibilityVoiceOver.
  2. Toggle VoiceOver to turn it on
  3. Navigate the site while listening to the announcements of the screen reader to understand if you have the full context of the application state.

VoiceOver

Testing workflow

  1. Keyboard navigation:

    1. Tab key: Use tab key to navigate through the interactive elements of the app and use Shift + Tab to go backwards.
    2. Ensure that you can navigate to each interactive element on the page.
  2. Navigating headings, buttons and links: In MacOS, you can use the VoiceOver rotor to directly access headings, links or main items. To navigate using rotor -

    1. Open rotor using VoiceOver modifier keys (Ctrl + Option) + U
    2. Use the rotor feature to navigate by headings, links, or form elements.
    3. Verify that headings and links convey the full context of the intended action.

    VoiceOver rotor VoiceOver rotor

    Other screen readers such as, JAWS and NVDA have similar features that display an elements list to allow users to quickly navigate between headings, links, lists, buttons, and more.

  3. Application keyboard shortcuts: Some applications, such as Slack, VSCode, provide their own keyboard shortcuts when navigating with assistive technologies in order to provide a better experience. When testing with screen reader, ensure you are able to use these shortcuts and have full context on where the focus is at.

Key Testing Areas

When testing with screen reader, it is helpful to consider how a user perceives the web page through the screen reader. Here’s an example of a user using a screen reader but having difficulty perceiving the meaning of the content due to poor accessibility development.

Watch a Screen Reader Demo for Digital Accessibility Video

When testing with a screen reader, here are a few things to pay attention to -

  1. Focus management:
    1. Ensure that when navigating with screen reader, focus is always known. It should be clear which element is focussed.
    2. Focus should move in a logical manner and should not haphazardly jump from one section to another.
    3. Manage focus appropriately and programmatically when needed, especially for modals, dialogues, or dynamic content, so users know where they are on the page.
    4. Return focus to a logical location after closing modals or completing actions.
  2. Label and role announcements: When navigating between components, ensure that elements are announced with their correct role and have a defined label. Here is a list of roles defined for HTML elements.
  3. Updates and notifications are announced: If there are urgent notifications delivered to the user, these notifications must be announced. An example is a user receiving a new message in Slack while browsing message history in Slack. This type of content, called dynamic content, is denoted by aria-live .

<div aria-live="polite">
<p> You have received 1 new message </p>
</div>

  1. Alt Text for Images: Ensure all images have descriptive alt text to convey their purpose or content. For purely decorative images, use an empty alt attribute (alt="") to hide them from screen readers. For images that are useful for the screen reader to know, it is important that the alt text gives enough context to the screen reader about the image. For example -

<img src="team-photo.jpg" alt="Team photo of employees at the company retreat" />

  1. Descriptive link labels: Links should provide clear context about their destination or purpose. Avoid vague labels like "click here" or "read more." Instead, use descriptive text.

<a href="/pricing">Click here</a> // Bad example
<a href="/pricing">View our pricing plans</a> // Good example

  1. Forms navigation:
    1. When navigating in a form, ensure that all fields are traversable and are announced with their proper label and type.
    2. Any errors, validation messages or form submission notifications must be announced.
    3. When multiple fields have errors, programmatically move focus to the first error field.
    4. Each form field must have a programmatically associated label.
    5. Provide clear error messages using aria-describedby for accessible feedback.

<label htmlFor="email">Email Address:</label>
<input id="email" type="email" aria-describedby="email-help" />
<span id="email-help" className="error-message">Please enter a valid email.</span>

  1. Clear Landmarks:

Use landmarks to provide structure and improve navigation. Screen readers allow users to jump between landmarks quickly through shortcuts, therefore providing landmarks enables efficient navigation.


<header>
<h1>Welcome to my Blog</h1>
</header>
<nav>
<ul>
<li><a href="/about">About Me</a></li>
<li><a href="/contact">Contact</a></li>
<li><a href="/blog">Blog</a></li>
</ul>
</nav>
<main>
<p>Main content goes here.</p>
</main>

Note that some landmarks should only appear on the page once - e.g., <h1> , <head> , <main> , etc.

Tips for ensuring a screen reader accessible site

  1. Keyboard-First Navigation: Interactions must be fully usable without a mouse, therefore prioritizing keyboard accessibility helps build a better experience for users of all abilities.

  2. Design with Accessibility in mind. Involve accessibility early. It is much easier to build an accessible app from the beginning instead of slapping it in later. Designing with accessibility in mind from the start helps to avoid retroactive fixes which can be extremely expensive.

  3. Avoid Visual-Only Cues

    1. Ensure important content is not conveyed solely through color, icons, or visual effects.
    2. Provide text alternatives or other cues (e.g., aria-hidden attributes for icons with text labels).
  4. Specify the lang attribute: Explicitly setting the lang attribute on the <html> tag ensures screen readers and other assistive technologies interpret the page's content in the correct language.


    <!DOCTYPE html>
    <html lang="en">
    <head>
    <title>Welcome Page</title>
    </head>
    <body>
    <h1>Welcome to My Blog </h1>
    </body>
    </html>

  5. Use semantic HTML: Using semantic HTML to convey the information of a block of element helps screen readers and assistive technologies. Overusing <div> and <span> elements, which have no semantic value, can cause screen readers to announce all of the content as a single block of text. However, remember to use semantic element for its appropriate functionality. Using an incorrect semantic tag can cause confusion and misrepresentation of UI elements.

  6. Avoid using abbreviations. Screen reader may not have context on abbreviations. If a screen reader encounters an abbreviation such as “CDC", it may not understand it and may pronounce it phonetically—which might sound like a jumble of consonants. To provide context, you can spell out the abbreviation, such as <abbr title="Centers for Disease Control and Prevention">CDC</abbr>. If the abbreviation is well-known — you might reverse this order, such as: HTML (HyperText Markup Language).

  7. Exercise caution when using aria-hidden.

    1. Never use aria-hidden on focusable elements. This might prevent users from perceiving important controls.
    2. Only hide content with ARIA when doing so will improve the user experience. For example, if an icon has an appropriate accessible name, you may hide the icon character, which would prevent screen readers from announcing the icon along with the accessible name.
  8. Content organization: Since a person using a screen reader might jump around your page looking for important information such as headings, bulleted lists, and links, it is helpful to organize content in a logical manner. Some tips to organize content are -

    1. Use ordered and unordered lists to break up longer content
    2. Use descriptive headings to organize ideas.
    3. Use headings in order. 
    4. Keep page layout consistent
  9. Automated Testing: Using an automated tool can help to highlight issues with semantic HTML, missing alt tags, inconsistent headings, etc. Use tools like:

  10. Axe DevTools

  11. Lighthouse

  12. WAVE

However, don’t rely solely on automated tools. Testing with real screen reader users is crucial for understanding the app’s user experience and identifying pain points that automated tools cannot highlight. Make sure to include real user testing in your process.

Conclusion

Testing with screen reader is an important part for ensuring accessible web apps. However, it is important to not rely on only one screen reader for testing. Different screen readers have different capabilities, and people use them in different ways. By ensuring that your app functions well with screen readers, we enable the web apps to be accessed by people with disabilities and people who are able bodied.

Remember, web accessibility isn’t just about screen readers. To learn out how to make web apps accessible, join <CTA> my workshop on building accessible web applications </CTA>

Shruti Kapoor profile picture

Join 40,000+ developers in the Epic Web community

Get the latest tutorials, articles, and announcements delivered to your inbox.

I respect your privacy. Unsubscribe at any time.