git clone https://github.com/NeverSight/learn-skills.dev
T=$(mktemp -d) && git clone --depth=1 https://github.com/NeverSight/learn-skills.dev "$T" && mkdir -p ~/.claude/skills && cp -r "$T/data/skills-md/agents-inc/skills/web-accessibility-web-accessibility" ~/.claude/skills/neversight-learn-skills-dev-web-accessibility-web-accessibility && rm -rf "$T"
data/skills-md/agents-inc/skills/web-accessibility-web-accessibility/SKILL.mdAccessibility
Quick Guide: All interactive elements keyboard accessible. Use headless component libraries for ARIA patterns. WCAG AA minimum (4.5:1 text contrast). Proper form labels and error handling. Combine automated axe-core checks with manual keyboard and screen reader testing.
Detailed Resources:
- examples/core.md - Skip links, semantic HTML, landmarks, button vs link
- examples/forms.md - Form validation, error handling, accessible select
- examples/focus.md - Modal dialogs, focus indicators
- examples/color.md - Contrast, color-independent indicators, tokens
- examples/tables.md - Sortable data tables
- examples/touch-targets.md - Touch target sizing
- examples/screen-reader.md - sr-only, hiding decorative content
- examples/testing.md - Accessibility testing with axe-core
- reference.md - Decision frameworks, anti-patterns, WCAG quick reference
<critical_requirements>
CRITICAL: Before Using This Skill
All code must follow project conventions in CLAUDE.md (kebab-case, named exports, import ordering,
, named constants)import type
(You MUST ensure all interactive elements are keyboard accessible with visible focus indicators)
(You MUST use headless component libraries for complex ARIA patterns instead of manual implementation)
(You MUST maintain WCAG AA minimum contrast ratios - 4.5:1 for text, 3:1 for UI components)
(You MUST never use color alone to convey information - always add icons, text, or patterns)
</critical_requirements>
Auto-detection: Accessibility (a11y), WCAG compliance, ARIA patterns, keyboard navigation, screen reader support, focus management, aria-label, aria-live, role attribute, skip link
When to use:
- Implementing keyboard navigation and focus management
- Ensuring WCAG AA color contrast (4.5:1 text, 3:1 UI components)
- Building interactive components (buttons, forms, modals, tables) with proper ARIA
- Adding dynamic content updates (live regions, status messages)
- Implementing skip links, landmarks, and semantic HTML structure
- Handling motion preferences with
prefers-reduced-motion
When NOT to use:
- Working on backend/API code with no UI
- Writing build scripts or configuration files
- Creating documentation or non-rendered content
- Working with CLI tools (different accessibility considerations)
Target: WCAG 2.2 Level AA compliance (minimum), AAA where feasible
<philosophy>
Philosophy
Accessibility ensures digital products are usable by everyone, including users with disabilities. Accessibility is a requirement, not a feature - it should be built in from the start, not retrofitted.
Key philosophy:
- Semantic HTML first - Use native elements for built-in accessibility
- Headless components for complex patterns - Leverage tested, accessible component primitives from your component library
- Progressive enhancement - Start with keyboard, add mouse interactions on top
- WCAG as baseline - Meet AA minimum, aim for AAA where feasible
<patterns>
Core Patterns
Keyboard Navigation Standards
CRITICAL: All interactive elements must be keyboard accessible
Tab Order
- Logical flow - Tab order must follow visual reading order (left-to-right, top-to-bottom)
- No keyboard traps - Users can always tab away from any element
- Skip repetitive content - Provide skip links to main content
- tabindex rules:
- Adds element to natural tab order (use sparingly)tabindex="0"
- Programmatic focus only (modal content, headings)tabindex="-1"- Never use
(creates unpredictable tab order)tabindex > 0
Focus Management
- Visible focus indicators - Always show clear focus state (never
without replacement)outline: none - Focus on open - When opening modals/dialogs, move focus to first interactive element or close button
- Focus on close - Restore focus to trigger element when closing modals/dialogs
- Focus trapping - Trap focus inside modals using your headless component library or manual implementation
- Programmatic focus - Use
for dynamic content (search results, error messages)element.focus()
Keyboard Shortcuts
- Standard patterns:
- Close modals, cancel actions, clear selectionsEscape
- Activate buttons and linksEnter/Space
- Navigate lists, tabs, menus, slidersArrow keys
- Jump to first/last itemHome/End
- Navigate between interactive elementsTab/Shift+Tab
Skip Links
MANDATORY for pages with navigation - place as first focusable element, visually hidden until focused.
// components/skip-link.tsx export function SkipLink({ className }: { className?: string }) { return ( <a href="#main-content" className={className}> Skip to main content </a> ); }
See examples/core.md for full skip link implementation with styling.
ARIA Patterns
Use headless component libraries - they handle ARIA automatically for complex patterns like dialogs, selects, tabs, tooltips, and popovers.
Component-Specific ARIA
Buttons:
- For icon-only buttonsaria-label
- For toggle buttonsaria-pressed
- For expandable sectionsaria-expanded
- Use witharia-disabled
attributedisabled
Forms:
- Required fields (use witharia-required
)required
- Invalid fieldsaria-invalid
- Link to error messages, helper textaria-describedby
- Explicit error message referencearia-errormessage
Navigation:
- Current page in navigationaria-current="page"
- Describe navigation purpose ("Main navigation", "Footer navigation")aria-label
Modals/Dialogs:
orrole="dialog"role="alertdialog"aria-modal="true"
- Points to dialog titlearia-labelledby
- Points to dialog descriptionaria-describedby
Tables:
andscope="col"
for headersscope="row"
for table description<caption>
for sortable columnsaria-sort
Live Regions
Use for dynamic content updates:
- Announce when user is idle (status messages, non-critical updates)aria-live="polite"
- Announce immediately (errors, critical alerts)aria-live="assertive"
- Announce entire region contentaria-atomic="true"
- For status messages (impliesrole="status"
)aria-live="polite"
- For error messages (impliesrole="alert"
)aria-live="assertive"
Best practices:
- Keep messages concise and meaningful
- Clear old messages before new ones
- Don't spam with rapid updates (debounce)
Color Contrast & Visual Design
Text contrast (AA):
- Normal text (< 18px): 4.5:1 minimum
- Large text (>= 18px or >= 14px bold): 3:1 minimum
- AAA (recommended): 7:1 for normal, 4.5:1 for large
Non-text contrast:
- UI components (buttons, form inputs): 3:1 minimum
- Focus indicators: 3:1 against background
- Icons (functional): 3:1 minimum
Color Independence:
- Add icons to color-coded states (check for success, X for error)
- Use text labels with status colors
- Provide patterns/textures in charts
- Underline links in body text
See examples/color.md for contrast examples and design tokens.
Semantic HTML
Always use semantic HTML:
for actions (not<button>
)<div onclick>
for navigation (not<a>
)<div onclick>
for navigation sections<nav>
for primary content (one per page)<main>
and<header>
for page sections<footer>
for lists<ul>/<ol>
for tabular data (not divs with grid CSS)<table>
with proper<form>
associations<label>
Never:
- Use
or<div>
for interactive elements<span> - Use click handlers on non-interactive elements without proper role
- Use tables for layout
- Use placeholder as label replacement
Form Accessibility
Label Associations:
- Always use proper
with<label>
attributehtmlFor - Or wrap input inside label element
Error Handling:
on invalid fieldsaria-invalid="true"
linking to error messagearia-describedby
on error messages for screen reader announcementrole="alert"- Visual error indicators (icons, border color)
- Error summary at top of form for multiple errors
See examples/forms.md for complete form validation and error handling patterns.
Focus Indicators
Minimum requirements:
- 3:1 contrast ratio against background
- 2px minimum thickness
- Clear visual difference from unfocused state
- Consistent across all interactive elements
Use
for better UX::focus-visible
/* Shows only for keyboard navigation, not mouse clicks */ button:focus-visible { outline: 2px solid var(--color-primary); outline-offset: 2px; }
See examples/focus.md for full focus indicator patterns.
Touch Target Sizes
WCAG 2.2 Target Size Requirements:
- Level AA (2.5.8): 24x24 CSS pixels minimum, or adequate spacing (24px between targets)
- Level AAA (2.5.5): 44x44 CSS pixels (recommended for best UX)
- Exceptions: inline text links, user agent controls, essential designs
See examples/touch-targets.md for sizing and spacing examples.
Screen Reader Support
- Use
class for screen reader only text.sr-only - Use
for icon-only buttonsaria-label - Use empty
for decorative imagesalt="" - Use
for decorative content (but never on focusable elements)aria-hidden="true"
See examples/screen-reader.md for sr-only and hiding patterns.
Motion and Animation Accessibility
WCAG 2.3.3 Animation from Interactions (AAA):
Motion can cause nausea, dizziness, or vestibular disorders (affects 70+ million people). Use
prefers-reduced-motion to respect user preferences:
/* Only animate when user has no motion preference */ @media (prefers-reduced-motion: no-preference) { .animated-element { animation: slide-in 300ms ease-out; } }
Key principles:
means minimize, not eliminate all motion - essential animations can remainreduce- Replace motion effects (scale, rotate, slide) with non-motion effects (fade, dissolve, color change)
- Provide pause/stop controls for auto-playing content longer than 5 seconds
- Avoid parallax scrolling effects or provide alternatives
WCAG 2.2 New Success Criteria
WCAG 2.2 (W3C Recommendation October 2023, ISO/IEC 40500:2025 as of October 2025) added 9 new success criteria. Key ones for developers:
Level A:
- 3.2.6 Consistent Help - Help mechanisms (contact, chat) must appear in same relative order across pages
- 3.3.7 Redundant Entry - Previously entered info must be auto-populated or available for selection
Level AA:
- 2.4.11 Focus Not Obscured (Minimum) - Focused element must not be entirely hidden by other content (sticky headers, modals)
- 2.5.7 Dragging Movements - Provide single-pointer alternative to drag operations
- 2.5.8 Target Size (Minimum) - 24x24px minimum or adequate spacing
- 3.3.8 Accessible Authentication - No cognitive function tests (CAPTCHAs) unless alternatives exist
Level AAA:
- 2.4.12 Focus Not Obscured (Enhanced) - No part of focus indicator hidden
- 2.4.13 Focus Appearance - Focus indicator 2px perimeter, 3:1 contrast
- 3.3.9 Accessible Authentication (Enhanced) - Stricter CAPTCHA requirements
Removed from WCAG 2.2:
- 4.1.1 Parsing - Obsolete due to modern browser error correction
Testing Accessibility
Multi-layered approach required - automated tools catch ~57% of WCAG issues. Always combine with manual testing.
Automated: Use axe-core based tools in your test runner for automated WCAG checks. See examples/testing.md for axe integration patterns.
Manual keyboard checklist:
- Tab through all interactive elements in logical order
- Activate buttons with Enter/Space
- Close modals with Escape
- Navigate dropdowns with arrows
- No keyboard traps
- Focus indicators visible on all elements
Manual screen reader checklist:
- All images have alt text (or alt="" if decorative)
- Form inputs have labels
- Error messages are announced
- Headings create logical outline
- Landmarks are labeled
- Live regions announce updates
Manual visual checklist:
- Color contrast meets WCAG AA (4.5:1 text, 3:1 UI)
- Information not conveyed by color alone
- Text resizable to 200% without horizontal scroll
- Touch targets meet 24x24px minimum (AA)
- Focus indicators have 3:1 contrast
- Animations respect prefers-reduced-motion
Screen readers to test with: NVDA (Windows, free), JAWS (Windows, paid), VoiceOver (macOS/iOS, built-in), TalkBack (Android, built-in)
<red_flags>
RED FLAGS
High Priority Issues:
- Removing focus outlines without replacement - keyboard users can't navigate, violates WCAG 2.4.7
- Using
ordiv
for buttons/links - no semantic meaning, no keyboard supportspan - Click handlers on non-interactive elements without role/keyboard support - violates WCAG 2.1.1
- Form inputs without labels - screen readers can't announce purpose, violates WCAG 1.3.1
Medium Priority Issues:
- Color-only error indicators - color-blind users can't distinguish
- Placeholder text as label replacement - disappears on input
- Auto-playing audio/video without controls - violates WCAG 1.4.2
Gotchas & Edge Cases:
vs:focus
- use:focus-visible
to avoid focus rings on mouse clicks:focus-visible- Empty
is correct for decorative images - don't skip the alt attribute entirelyalt=""
also hides from keyboard - don't use on focusable elementsaria-hidden="true"
onrole="button"
doesn't add keyboard support - still need Enter/Space handlers<div>
means minimize, not eliminate - essential animations can remainprefers-reduced-motion: reduce- Live regions announce ALL content - keep messages concise to avoid spam
See reference.md for full anti-patterns with code examples.
</red_flags>
Resources
Official guidelines:
- WCAG 2.2 Guidelines: https://www.w3.org/WAI/WCAG22/quickref/
- What's New in WCAG 2.2: https://www.w3.org/WAI/standards-guidelines/wcag/new-in-22/
- WAI-ARIA Authoring Practices: https://www.w3.org/WAI/ARIA/apg/
Tools:
- axe DevTools: https://www.deque.com/axe/devtools/
- WAVE: https://wave.webaim.org/
- WebAIM Contrast Checker: https://webaim.org/resources/contrastchecker/
<critical_reminders>
CRITICAL REMINDERS
All code must follow project conventions in CLAUDE.md (kebab-case, named exports, import ordering,
, named constants)import type
(You MUST ensure all interactive elements are keyboard accessible with visible focus indicators)
(You MUST use headless component libraries for complex ARIA patterns instead of manual implementation)
(You MUST maintain WCAG AA minimum contrast ratios - 4.5:1 for text, 3:1 for UI components)
(You MUST never use color alone to convey information - always add icons, text, or patterns)
Failure to follow these rules will make the site unusable for keyboard users, screen reader users, and color-blind users - violating WCAG 2.2 Level AA compliance.
</critical_reminders>