Accessibility Testing
Accessibility Testing
This skill enables the agent to perform thorough accessibility audits of web pages and components against the Web Content Accessibility Guidelines (WCAG) 2.1 at AA and AAA conformance levels. The agent identifies violations across four principles — Perceivable, Operable, Understandable, Robust — and generates structured compliance reports with specific code fixes. It covers automated checks (color contrast, missing alt text, ARIA misuse), semi-automated checks (keyboard navigation flows, focus management), and manual check guidance (screen reader announcements, cognitive load).
Workflow
-
Define Audit Scope and Conformance Target: Determine which pages, components, or user flows to audit, and whether the target is WCAG 2.1 AA (most common legal requirement) or AAA (highest conformance). Identify the assistive technologies to consider: screen readers (NVDA on Windows, VoiceOver on macOS/iOS, TalkBack on Android), keyboard-only navigation, and magnification tools.
-
Run Automated Scans: Execute automated accessibility analysis using axe-core or Lighthouse. These tools detect approximately 30-40% of WCAG violations automatically, including missing alt attributes, insufficient color contrast ratios, missing form labels, duplicate IDs, invalid ARIA roles, and missing document language. Capture each violation with its WCAG criterion reference (e.g., 1.4.3 Contrast Minimum), severity (critical, serious, moderate, minor), the affected HTML element, and the CSS selector path.
-
Perform Keyboard Navigation Testing: Manually trace every interactive flow using only Tab, Shift+Tab, Enter, Space, Escape, and arrow keys. Verify that focus order matches visual order, that focus indicators are visible (minimum 2px outline with 3:1 contrast), that modal dialogs trap focus correctly, and that no keyboard traps exist. Document any flow where a keyboard user cannot complete a task that a mouse user can.
-
Test with Screen Readers: Navigate the page with at least one screen reader to verify that all content is announced correctly. Confirm that images have meaningful alt text (or
alt=""for decorative images), that headings form a logical hierarchy (no skipped levels), that landmarks (<nav>,<main>,<aside>) are present, and that dynamic content updates are announced viaaria-liveregions. -
Generate the Compliance Report: Produce a structured report organized by WCAG principle and criterion. For each violation, include: the criterion number and name, the severity level, the affected element's HTML snippet, an explanation of why it fails, and the specific code fix. Include a summary table with pass/fail counts per principle and an overall conformance statement.
-
Provide Remediation Code: For every violation, supply a concrete before-and-after code snippet showing the exact fix. Group fixes by component or page region so developers can address them systematically rather than one element at a time.
Usage
Provide the agent with a URL, a block of HTML, or a React/Vue component. Specify the target conformance level (AA or AAA) and any assistive technology requirements. The agent will audit the content and return a structured report with code fixes.
Examples
Example 1: WCAG Compliance Audit Report
User Request: "Audit this login form for WCAG 2.1 AA compliance."
<div class="login">
<img src="logo.png">
<input type="text" placeholder="Username">
<input type="password" placeholder="Password">
<div class="btn" onclick="login()">Sign In</div>
<p style="color: #aaa; font-size: 12px;">Forgot your password?</p>
</div>
Agent Response — Accessibility Audit Report:
| # | Criterion | Severity | Element | Issue |
|---|---|---|---|---|
| 1 | 1.1.1 Non-text Content | Critical | <img src="logo.png"> |
Image has no alt attribute. Screen readers cannot describe it. |
| 2 | 1.3.1 Info and Relationships | Critical | <input placeholder="Username"> |
Input has no associated <label>. Placeholder text is not a substitute for a label. |
| 3 | 1.3.1 Info and Relationships | Critical | <input placeholder="Password"> |
Same label issue as above. |
| 4 | 1.4.3 Contrast Minimum | Serious | <p style="color: #aaa"> |
Text color #aaa on white background has a contrast ratio of 2.32:1. AA requires 4.5:1 for text this size. |
| 5 | 2.1.1 Keyboard | Critical | <div class="btn" onclick> |
A <div> is not focusable via keyboard. Users cannot Tab to or activate this button. |
| 6 | 4.1.2 Name, Role, Value | Critical | <div class="btn" onclick> |
Element has no semantic role. Screen readers announce it as generic text, not as a button. |
Remediation — Fixed Code:
<form class="login" onsubmit="login(event)">
<img src="logo.png" alt="Acme Corp logo">
<label for="username">Username</label>
<input id="username" type="text" autocomplete="username" required>
<label for="password">Password</label>
<input id="password" type="password" autocomplete="current-password" required>
<button type="submit" class="btn">Sign In</button>
<a href="/forgot-password" style="color: #595959; font-size: 14px;">
Forgot your password?
</a>
</form>
Changes made: added alt to the image, replaced placeholder-only inputs with explicit <label> elements, replaced the <div> with a semantic <button>, changed the link color from #aaa (2.32:1) to #595959 (7.0:1), wrapped the form in a <form> element for proper submit handling, and added autocomplete attributes.
Example 2: Fixing a React Component
User Request: "Fix accessibility issues in this React notification component."
Before (inaccessible):
function Notification({ message, onClose }) {
return (
<div className="notification">
<span>{message}</span>
<span className="close-x" onClick={onClose}>✕</span>
</div>
);
}
After (accessible):
function Notification({ message, onClose }) {
return (
<div role="alert" aria-live="assertive" className="notification">
<p>{message}</p>
<button
type="button"
onClick={onClose}
aria-label="Dismiss notification"
className="close-btn"
>
✕
</button>
</div>
);
}
Fixes applied: (1) Added role="alert" and aria-live="assertive" so screen readers announce the notification immediately when it appears. (2) Replaced the <span onClick> with a <button> so it is keyboard-focusable and announced as an interactive control. (3) Added aria-label="Dismiss notification" because the "✕" character alone does not convey the button's purpose to screen reader users. (4) Changed the inner <span> to a <p> for proper text semantics.
Best Practices
- Run automated tools first, but never rely on them alone: axe-core catches roughly 30-40% of WCAG issues. The remaining 60-70% require manual keyboard testing, screen reader verification, and cognitive review.
- Test with real screen readers, not just ARIA validators: An element may have correct ARIA markup but still produce confusing announcements. VoiceOver and NVDA sometimes interpret the same markup differently.
- Fix critical and serious issues before moderate ones: Prioritize violations that completely block access (missing keyboard operability, no alt text on functional images) over cosmetic issues (minor contrast shortfalls on decorative elements).
- Use semantic HTML before reaching for ARIA: A
<button>needs norole="button". A<nav>needs norole="navigation". ARIA is a repair tool for situations where semantic HTML is not sufficient, not a replacement for it. - Include accessibility checks in CI pipelines: Run axe-core or pa11y in automated tests so new violations are caught before they reach production. Fail the build on critical violations.
- Document accessibility decisions: When a component intentionally deviates from a guideline (e.g., a custom combobox pattern), document the rationale and the alternative approach used to maintain equivalent access.
Edge Cases
- Single-page applications with client-side routing: Page navigation does not trigger a browser page load, so screen readers are not notified of the new content. Use an
aria-live="polite"region to announce route changes, or programmatically move focus to the new page's<h1>. - Dynamic content loaded after initial render: Content injected via JavaScript after page load is invisible to screen readers unless wrapped in an
aria-liveregion or focus is explicitly managed. For toast notifications usearia-live="assertive"; for feed updates usearia-live="polite". - Complex data tables: Tables with merged cells, nested headers, or sortable columns require explicit
scope,headers, andaria-sortattributes. Test that a screen reader user can understand which header applies to each data cell. - Custom interactive widgets (sliders, date pickers, comboboxes): These have no native HTML equivalent. Follow the WAI-ARIA Authoring Practices 1.2 patterns exactly, implementing the full keyboard interaction model specified for each widget type.
- Third-party embedded content (iframes, widgets): You cannot fix accessibility inside a third-party iframe. Document the issue, add a descriptive
titleattribute to the<iframe>, and provide an accessible alternative when the embedded content is critical to the user flow.
More from seb1n/awesome-ai-agent-skills
summarization
Summarize text using extractive, abstractive, hierarchical, and multi-document techniques, producing concise outputs at configurable detail levels.
23note-taking
Capture, organize, and retrieve notes efficiently using structured formats, tagging, and file management for meetings, ideas, research, and daily logs.
18knowledge-graph-creation
Build structured knowledge graphs from unstructured text by extracting entities, mapping relationships, generating graph triples, and visualizing the result.
16proofreading
Proofread and correct text for grammar, spelling, punctuation, style, clarity, and consistency, with support for multiple style guides and readability analysis.
16technical-writing
Write clear, concise, and accurate technical documentation including API references, user guides, tutorials, changelogs, and architecture docs, tailored to the target audience.
14data-analysis
Analyze datasets to extract insights through statistical methods, trend identification, hypothesis testing, and correlation analysis.
14