Accessibility Skill
Step 1: Semantic HTML Audit
Review component structure for proper semantic elements:
Check for:
-
<header> , <nav> , <main> , <article> , <section> , <aside> , <footer> instead of generic <div>
-
<button> for clickable elements (not <div onclick> )
-
<a> for navigation links
-
<form> , <input> , <label> for forms
-
Proper heading hierarchy (<h1> through <h6> )
Example:
<!-- ❌ BAD --> <div class="header"> <div claass="nav"> <div class="nav-item" onclick="navigate()">Home</div> </div> </div>
<!-- ✅ GOOD --> <header> <nav> <a href="/">Home</a> </nav> </header>
Step 2: ARIA Attributes Review
Add ARIA attributess ONLY when semantic HTML is insufficient:
Common Patterns:
Use Case ARIA Attributes Example
Custom button role="button" , tabindex="0"
<div role="button" tabindex="0">
Modal dialog role="dialog" , aria-modal="true"
<div role="dialog" aria-modal="true">
Alert role="alert" , aria-live="assertive"
<div role="alert">Error occurred</div>
Tab panel role="tabpanel" , aria-labelledby
<div role="tabpanel" aria-labelledby="tab-1">
Rules:
-
Don't add redundant ARIA (<button role="button"> is unnecessary)
-
Use aria-label for icon buttons without text
-
Use aria-hidden="true" for decorative elements
-
Use aria-live regions for dynamic content
Example:
<!-- Icon button needs aria-label --> <button aria-label="Close dialog"> <i class="icon-close" aria-hidden="true"></i> </button>
<!-- Dynamic content needs live region --> <div role="alert" aria-live="assertive">Form submitted successfully</div>
Step 3: Keyboard Navigation Test
Verify all interactive elements are keyboard accessible:
Requirements:
-
Tab: Navigate forward through interactive elements
-
Shift+Tab: Navigate backward
-
Enter/Space: Activate buttons and links
-
Arrow keys: Navigate within components (tabs, menus, listboxes)
-
Escape: Close dialogs and menus
Focus Management:
-
Trap focus within modals (prevent tabbing outside)
-
Return focus to trigger element when closing modal
-
Skip to main content link for screen reader users
-
Visible focus indicators (:focus styles)
Example:
// Focus trap in modal function openModal(modal) { modal.style.display = 'block'; const firstFocusable = modal.querySelector('button, input, a'); firstFocusable.focus(); trapFocus(modal); // Prevent escape from modal }
function trapFocus(container) { const focusableElements = container.querySelectorAll('button, input, select, textarea, a[href]'); const firstElement = focusableElements[0]; const lastElement = focusableElements[focusableElements.length - 1];
container.addEventListener('keydown', e => { if (e.key === 'Tab') { if (e.shiftKey && document.activeElement === firstElement) { lastElement.focus(); e.preventDefault(); } else if (!e.shiftKey && document.activeElement === lastElement) { firstElement.focus(); e.preventDefault(); } } }); }
Step 3B: WCAG 2.2 New Success Criteria (October 2023, ISO/IEC 40500:2025)
WCAG 2.2 added 9 new success criteria. Verify compliance with the applicable ones:
2.4.11 Focus Not Obscured — Minimum (AA)
When a keyboard-focused element scrolls into view, it must not be completely hidden by sticky headers/footers or overlapping UI. At least part of the focused element must always be visible.
/* Prevent focus from hiding behind sticky headers / :focus-visible { scroll-margin-top: 80px; / Height of sticky header + buffer / scroll-margin-bottom: 60px; / Height of sticky footer + buffer */ }
Test: Tab through all interactive elements — verify none are fully hidden behind banners, cookie notices, or sticky bars.
2.4.12 Focus Not Obscured — Enhanced (AAA)
The focused component must be fully visible (not just partially). Sticky UI must not overlap focus at all.
2.4.13 Focus Appearance (AAA)
Focus indicators must have: area of at least the perimeter of the unfocused component times 2 CSS pixels, and contrast ratio of at least 3:1 between focused and unfocused states.
/* Meeting 2.4.13 Focus Appearance (AAA) / :focus-visible { outline: 3px solid #005fcc; outline-offset: 2px; / 3px thickness > 2px minimum; #005fcc on white = 7.1:1 contrast */ }
2.5.7 Dragging Movements (AA)
All drag-and-drop functionality MUST have a single-pointer (click/tap) alternative. Users who cannot perform precise drag gestures must be able to accomplish the same task.
<!-- Sortable list: drag alternative via buttons --> <ul> <li draggable="true" id="item-1"> Item 1 <button aria-label="Move Item 1 up" onclick="moveUp('item-1')">↑</button> <button aria-label="Move Item 1 down" onclick="moveDown('item-1')">↓</button> </li> </ul>
<!-- Slider: keyboard alternative provided natively --> <input type="range" min="0" max="100" value="50" aria-label="Volume" /> <!-- Alternatively: add an input[type=number] companion --> <input type="number" min="0" max="100" value="50" aria-label="Volume value" />
Test: Identify all drag interactions. Verify each has a non-drag equivalent (buttons, context menus, keyboard shortcuts).
2.5.8 Target Size — Minimum (AA)
All pointer targets (buttons, links, checkboxes, radio inputs) must be at least 24x24 CSS pixels, OR have sufficient spacing so the 24x24 area does not overlap another target.
/* Ensure minimum target size */ button, a, input[type='checkbox'], input[type='radio'], select { min-width: 24px; min-height: 24px; }
/* Inline links: use padding to increase hit area without visual size change */ a { padding: 4px 0; }
/* Recommended: 44x44 CSS pixels for primary actions (mobile-friendly) */ .primary-action { min-width: 44px; min-height: 44px; }
Test: Measure all interactive elements. Verify none fall below 24x24 CSS pixels (use browser DevTools element inspector).
3.2.6 Consistent Help (A)
If a help mechanism (contact link, chat widget, phone number, FAQ link) appears on multiple pages, it must appear in the same relative order in the page content.
<!-- Help mechanism must appear consistently across pages --> <footer> <nav aria-label="Help resources"> <!-- This order must not change between pages --> <a href="/faq">FAQ</a> <a href="/contact">Contact Support</a> <a href="tel:+18005551234">1-800-555-1234</a> </nav> </footer>
Test: Navigate between pages. Verify help links/widgets appear in the same order each time.
3.3.7 Redundant Entry (A)
Information previously entered by the user must be auto-populated or available for selection when the same information is requested again in the same session/process (e.g., multi-step checkout forms).
<!-- Step 2: Billing address — pre-fill from Step 1 shipping --> <fieldset> <legend>Billing Address</legend> <label> <input type="checkbox" id="same-as-shipping" /> Same as shipping address </label> <!-- When checked: auto-populate billing fields from shipping fields --> </fieldset>
Exceptions: Re-entering passwords for security confirmation, selecting items from a list.
3.3.8 Accessible Authentication — Minimum (AA)
Authentication processes MUST NOT require users to complete a cognitive function test (solve puzzle, identify images, remember/transcribe a code) unless an accessible alternative is provided.
Allowed alternatives:
-
Biometric authentication (fingerprint, face recognition)
-
Email magic link / SMS OTP (user does not need to recall the code — just copy-paste)
-
OAuth via a third-party provider
-
Password managers allowed (do not block paste in password fields)
<!-- GOOD: Allow paste in password fields --> <input type="password" id="password" autocomplete="current-password" /> <!-- Do NOT add: onpaste="return false" -->
<!-- GOOD: Provide alternative to image CAPTCHA --> <div role="group" aria-labelledby="captcha-label"> <span id="captcha-label">Verify you are human</span> <img src="captcha.png" alt="CAPTCHA challenge" /> <input type="text" aria-describedby="captcha-label" /> <a href="?audio-captcha">Use audio CAPTCHA instead</a> <a href="?email-login">Use email link instead</a> </div>
Test: Identify all authentication steps. Verify no step requires a cognitive test without an alternative method.
3.3.9 Accessible Authentication — No Exception (AAA)
No cognitive function test is required, even with alternatives provided.
Step 4: Color Contrast Verification
Check all text meets WCAG contrast ratios:
Standards:
Text Size WCAG AA WCAG AAA
Normal text (< 18pt) 4.5:1 7:1
Large text (≥ 18pt or 14pt bold) 3:1 4.5:1
UI components 3:1
Tools:
-
WebAIM Contrast Checker
-
Browser DevTools color picker
-
Grayscale test (convert to grayscale to verify readability)
Example:
/* ❌ BAD - Innsufficient contrast / .text { color: #777; background: #fff; } / 4.47:1 - fails AA */
/* ✅ GOOD - Sufficient contrast / .text { color: #595959; background: #fff; } / 7:1 - passes AAA */
/* ✅ GOOD - Don't rely on color alone / .error { color: #d00; border-left: 4px solid #d00; / Visual indicator beyond color / } .error::before { content: '⚠️ '; } / Icon indicator */
Step 5: Screen Reader Support
Ensure proper sscreen reader experience:
Alt Text for Images:
<!-- ❌ BAD - Missing or redundant alt --> <img src="logo.png" /> <img src="decorative.png" alt="decorative image" />
<!-- ✅ GOOD --> <img src="logo.png" alt="Company y Logo" /> <img src="decorative.png" alt="" role="presentation" />
ARIA Labels for Icon Buttons:
<!-- ❌ BAD - No label for screen readers --> <button><i class="icon-delete"></i></button>
<!-- ✅ GOOD --> <but tton aria-label="Delete item"> <i class="icon-delete" aria-hidden="true"></i> </button>
Live Regions for Dynamic Content:
<!-- Announce errors immediately --> <div role="alert" aria-live="assertive">Error: Invalid email address</div>
<!-- Announce status updates politely --> <div aria-live="polite" aria-atomic="true">Loading results... 3 of 10 loaded</div>
Step 6: Form Accessibility
Ensure all form inputs are properly labeled and validated:
Requirements:
-
All inputs have associated <label> elements
-
Use <fieldset> and <legend> for grouped inputs
-
Show validation errors with aria-describedby
-
Required fields marked with aria-required="true" or required attribute
Example:
<!-- ✅ GOOD Form Structure --> <form> <fieldset> <legend>Personal Information</legend>
<label for="name">Name (required)</label>
<input id="name" type="textt" required aria-required="true" aria-describedby="name-error" />
<span id="name-error" role="alert" class="error" aria-live="polite">
<!-- Error message appears here -->
</span>
<label for="email">Email</label>
<input id="email" type="email" aria-describedby="email-hint" />
<span id="email-hint" class="hint">We'll never share your email</span>
</fieldset> </form>
Step 7: Generate Accessibility Report
Document findings with:
-
Total issues found (categorized by severity)
-
WCAG level compliance status (A, AA, AAA)
-
Specific violations with line numbers
-
Recommended fixes with code examples
-
Testing performed (automated + manual)
Example 1: Review React Component
Skill({ skill: 'accessibility' });
Input: React component with custom modal Output:
-
Semantic HTML recommendations
-
ARIA attributes needed
-
Keyboard navigation issues
-
Focus trap implementation
-
WCAG compliance report
Example 2: Audit Color Contrast
Skill({ skill: 'accessibility', args: 'color-contrast' });
Input: CSS file with color definitions Output:
-
List of failing contrast ratios
-
Recommended color adjustments
-
Before/after contrast scores
Example 3: Form Accessibility Check
Skill({ skill: 'accessibility', args: 'forms' });
Input: Form component Output:
-
Label associations verified
-
Required field indicators
-
Error message patterns
-
Keyboard submission support
<best_practices>
Best Practices
DO
-
Use semantic HTML as foundation (header, nav, main, article)
-
Add ARIA only when semantic HTML insufficient
-
Test with real screen readers (NVDA, JAWS, VoiceOver)
-
Ensure keyboard navigation works without mouse
-
Maintain 4.5:1 contrast for normal text (WCAG AA)
-
Provide text alternatives for all non-text content
-
Use focus indicators (visible :focus-visible styles)
-
Trap focus within modals
-
Announce dynamic content with aria-live
-
Ensure focused elements are not obscured by sticky headers/footers (WCAG 2.2 2.4.11)
-
Provide keyboard/click alternatives for all drag interactions (WCAG 2.2 2.5.7)
-
Size all pointer targets to at least 24x24 CSS pixels (WCAG 2.2 2.5.8)
-
Allow paste in password fields — never block it (WCAG 2.2 3.3.8)
-
Auto-populate previously entered data in multi-step processes (WCAG 2.2 3.3.7)
-
Place help mechanisms in consistent order across pages (WCAG 2.2 3.2.6)
DON'T
-
Use <div> for everything (no semantic meaning)
-
Put click handlers on non-interactive elements
-
Forget alt text on images
-
Rely on color alone for information
-
Remove focus indicators (outline: none)
-
Auto-play media without controls
-
Use tabindex
0 (disrupts natural tab order)
-
Create keyboard traps (user can't escape)
-
Hide important content from screen readers
Anti-Patterns
Anti-Pattern Problem Fix
<div onclick>
Not keyboard accessible Use <button>
No alt text Screen readers can't describe Add meaningful alt attribute
Color-only info Color blind users miss it Add text/icons
No focus indicators Users lost in navigation Add :focus-visible styles
Auto-play media Disruptive for screen readers Add controls, pause option
<div> for everything No semantic structure Use semantic HTML
Sticky header without scroll-margin Focus hidden behind sticky bar (2.4.11) Add scroll-margin-top to :focus-visible
Drag-only sortable lists Users with motor disabilities can't sort (2.5.7) Add Up/Down buttons for each item
16x16px icon buttons Below 24x24 minimum target size (2.5.8) Set min-width: 24px; min-height: 24px
CAPTCHA without alternative Cognitive barrier to authentication (3.3.8) Provide email magic link or passkey option
onpaste="return false"
Blocks password manager paste (3.3.8) Remove paste block from password fields
Repeated form fields in checkout Redundant data entry (3.3.7) Auto-populate with previously entered values
Testing Checklist
Before finalizing accessibility review:
WCAG 2.1 AA (existing requirements):
-
All images have alt text (or alt="" for decorative)
-
All interactive elements keyboard accessible
-
Tab order is logical
-
Focus indicators visible
-
Color contrast meets WCAG AA (4.5:1 normal, 3:1 large)
-
Semantic HTML used (nav, main, article, etc.)
-
ARIA labels on icon buttons
-
Forms have proper labels
-
Error messages announced to screen readers
-
Dialogs trap focus and close on Escape
-
Dynamic content uses ARIA live regions
-
Tested with screen reader (NVDA, JAWS, VoiceOver)
-
Tested with keyboard only (no mouse)
-
Tested with browser zoom (200%)
WCAG 2.2 AA (new requirements — October 2023, ISO/IEC 40500:2025):
-
2.4.11: Focused elements not completely obscured by sticky headers/footers
-
2.5.7: All drag interactions have a single-pointer (click/tap) alternative
-
2.5.8: All pointer targets are at least 24x24 CSS pixels
-
3.2.6: Help mechanisms (chat, contact, phone) appear in same order across pages
-
3.3.7: Previously entered information is auto-populated in multi-step forms
-
3.3.8: Authentication does not require cognitive function test (puzzle/CAPTCHA) without alternative
-
Password fields allow paste (do not block paste with onpaste="return false")
-
Focus scroll margin set to prevent sticky UI from hiding keyboard focus </best_practices>
Iron Laws
-
ALWAYS start with semantic HTML — never reach for ARIA before using the right native element (<button> , <nav> , <main> , etc.).
-
NEVER remove focus indicators — outline: none without a replacement is an immediate WCAG failure. Keyboard users become completely lost.
-
ALWAYS test with real assistive technology — automated tools (axe, Lighthouse) catch at most 30% of issues. NVDA, JAWS, or VoiceOver testing is mandatory.
-
NEVER convey information by color alone — always pair color with text, icons, or patterns for users with color vision deficiencies.
-
ALWAYS apply WCAG 2.2 AA criteria — 2.4.11 (focus not obscured), 2.5.7 (drag alternatives), 2.5.8 (24×24 target size), 3.3.8 (no cognitive auth barriers) are mandatory, not optional.
Integration Points
Agents Using This Skill
-
developer: Implements accessible components
-
code-reviewer: Reviews accessibility in PRs
-
qa: Tests accessibility compliance
-
frontend-pro: Ensures accessible UI patterns
-
react-pro: React-specific accessibility patterns
Related Skills
-
frontend-expert: UI component patterns
-
react-expert: React accessibility patterns
-
mobile-first-design-rules: Touch accessibility
Workflows
-
feature-development-workflow.md: Accessibility review in Review phase
-
code-review-workflow.md: Accessibility checklist
Related References
-
.claude/rules/accessibility.md
-
Complete accessibility rules
-
WCAG 2.2 Guidelines (current standard)
-
What's New in WCAG 2.2
-
WCAG 2.2 Quick Reference
-
WebAIM Contrast Checker
-
ARIA Authoring Practices (APG)
-
WCAG 2.2 Complete Implementation Guide (TestParty)
-
European Accessibility Act compliance
Memory Protocol (MANDATORY)
Before starting:
cat .claude/context/memory/learnings.md
Check for:
-
Previously discovered accessibility patterns
-
Common accessibility issues in this codebase
-
Project-specific accessibility requirements
After completing:
-
New accessibility pattern → .claude/context/memory/learnings.md
-
Accessibility issue found → .claude/context/memory/issues.md
-
Accessibility decision made → .claude/context/memory/decisions.md
ASSUME INTERRUPTION: Your context may reset. If it's not in memory, it didn't happen.