Use this skill for any accessibility concern: WCAG compliance (any success criterion including SC 1.3.5 autocomplete, SC 2.4.11 focus obscured), ARIA roles/attributes, screen reader behavior, axe-core or WAVE audit failures, color/non-text contrast, keyboard navigation, focus trapping/management, skip links, touch targets, VPAT reports, or accessible form patterns. Also trigger when a user asks whether a UI library (DSAI) handles keyboard interactions — but only if the question is specifically about accessibility behavior, not general usage. Do NOT trigger for general component usage questions where keyboard behavior is incidental (e.g., "does DSAI Tooltip show on hover and focus?" without an accessibility problem to solve).
From hugin-v0npx claudepluginhub michelve/hugin-marketplace --plugin hugin-v0This skill uses the workspace's default tool permissions.
LICENSE.txtevals/evals.jsonevals/run_loop_results/2026-03-12_103908/report.htmlevals/run_loop_results/2026-03-12_103908/results.jsonevals/run_loop_results/2026-03-12_104112/report.htmlevals/run_loop_results/2026-03-12_104112/results.jsonevals/run_loop_results/2026-03-12_105717/logs/improve_iter_1.jsonevals/run_loop_results/2026-03-12_105717/logs/improve_iter_2.jsonevals/run_loop_results/2026-03-12_105717/logs/improve_iter_3.jsonevals/run_loop_results/2026-03-12_105717/logs/improve_iter_4.jsonevals/run_loop_results/2026-03-12_105717/report.htmlevals/run_loop_results/2026-03-12_105717/results.jsonevals/run_loop_results/2026-03-12_111808/logs/improve_iter_1.jsonevals/run_loop_results/2026-03-12_111808/logs/improve_iter_2.jsonevals/run_loop_results/2026-03-12_111808/logs/improve_iter_3.jsonevals/run_loop_results/2026-03-12_111808/logs/improve_iter_4.jsonevals/run_loop_results/2026-03-12_111808/report.htmlevals/run_loop_results/2026-03-12_111808/results.jsonevals/run_loop_results/2026-03-12_112056/logs/improve_iter_1.jsonevals/run_loop_results/2026-03-12_112056/report.htmlGuides browser automation with Playwright, Puppeteer, Selenium for e2e testing and scraping. Teaches reliable selectors, auto-waits, isolation to fix flaky tests.
Provides checklists to review code for functionality, quality, security, performance, tests, and maintainability. Use for PRs, audits, team standards, and developer training.
Enforces A/B test setup with gates for hypothesis locking, metrics definition, sample size calculation, assumptions checks, and execution readiness before implementation.
Standard: WCAG 2.2 AA required · AAA aspirational Stack: React 19 · DSAI Design System · Bootstrap 5 · CSS Custom Properties (--dsai-* tokens) POUR Principles: Perceivable · Operable · Understandable · Robust
Implement WCAG 2.2 AA compliant accessibility in React 19 components using the draft_v0 stack. Items marked (AAA) exceed the required AA level and are aspirational best practices to pursue where feasible.
| Principle | Key Topics | Reference Files |
|---|---|---|
| Perceivable | Alt text, landmarks, headings, data tables, color contrast, non-text contrast, text spacing, reflow | semantic-html.md, mobile-touch.md |
| Operable | Keyboard, focus management, focus not obscured, skip links, page titles, target size, dragging, pointer gestures | keyboard-navigation.md, focus-management.md, mobile-touch.md, wcag22-new-criteria.md |
| Understandable | ARIA, forms, error handling, auth, redundant entry, consistent help, live regions | aria-attributes.md, forms-a11y.md |
| Robust | Screen reader support, status messages, automated + manual testing, VPAT, CI/CD | testing-a11y.md |
WCAG 2.2 added 9 success criteria not present in WCAG 2.1. All AA items are required; AAA items are aspirational.
| SC | Title | Level | Requirement in One Line |
|---|---|---|---|
| 2.4.11 | Focus Not Obscured (Minimum) | AA | Focused element not fully hidden behind sticky/fixed UI chrome |
| 2.4.12 | Focus Not Obscured (Enhanced) | AAA | Focused element not obscured at all — not even partially |
| 2.4.13 | Focus Appearance | AAA | Focus indicator ≥ 2px perimeter, 3:1 contrast ratio |
| 2.5.7 | Dragging Movements | AA | Every drag action has a single-pointer (click/tap) alternative |
| 2.5.8 | Target Size (Minimum) | AA | Interactive targets ≥ 24×24 CSS pixels |
| 3.2.6 | Consistent Help | A | Help mechanisms in same relative position across pages |
| 3.3.7 | Redundant Entry | A | Previously entered data auto-populated or selectable |
| 3.3.8 | Accessible Authentication (Minimum) | AA | No cognitive puzzle required in auth unless alternative exists |
| 3.3.9 | Accessible Authentication (Enhanced) | AAA | No cognitive test in any auth step, no exceptions |
See wcag22-new-criteria.md for detailed implementation guidance, code examples, common failures, and test procedures for each criterion.
| Tool / Pattern | How to Use | Notes |
|---|---|---|
useFocusTrap | import { useFocusTrap } from '@/hooks' | DSAI hook for focus trapping in modals/dialogs. |
useRovingFocus | import { useRovingFocus } from '@/hooks' | Arrow key navigation within tab lists, menus. |
useReducedMotion | import { useReducedMotion } from '@/hooks' | Respects prefers-reduced-motion OS preference. |
visually-hidden | className="visually-hidden" | Bootstrap class — visually hidden, screen-reader visible text. |
useKeyPress | import { useKeyPress } from '@/hooks' | Keyboard event handling hook. |
useScrollLock | import { useScrollLock } from '@/hooks' | Prevents body scroll when modal/sheet is open. |
SafeHTMLAttributes | import type { SafeHTMLAttributes } from '@/types' | Whitelisted HTML attributes — blocks event handlers in spread. |
announceToScreenReader | import { announceToScreenReader } from '@/utils' | DSAI utility for live region announcements. |
| Custom hooks | src/client/hooks/ (@/hooks) | useFocusTrap, useKeyPress, useReducedMotion, useRovingFocus, useScrollLock, useHover. |
DSAI components implement accessibility patterns directly — using SafeHTMLAttributes, ARIA props, keyboard hooks, and focus management. Components handle ARIA roles, keyboard navigation, and focus management through the DSAI hook system.
| Component | What DSAI Handles |
|---|---|
Modal | Focus trap (useFocusTrap), scroll lock (useScrollLock), Escape to close, return focus on close, aria-modal |
Dropdown | role="menu" / menuitem, roving focus (useRovingFocus), Arrow keys, Escape to close |
Select | role="listbox" / option, Arrow keys, type-ahead search |
Tabs | role="tablist" / tab / tabpanel, Arrow Left/Right (useRovingFocus), activation modes |
Checkbox | role="checkbox", Space to toggle, aria-checked |
Tooltip | role="tooltip", hoverable, dismissible with Escape, persistent |
Popover | Focus-managed popup, Escape to close, returns focus to trigger |
Accordion | aria-expanded, Arrow keys for panel navigation |
Switch | role="switch", aria-checked, Space/Enter to toggle |
Button | FSM-based states (loading, error, disabled), announceText for screen readers |
Install via: dsai add modal tabs select — components appear in src/client/components/ui/ as local source code you own and can customize.
Use native HTML5 elements — they carry implicit ARIA roles and need no role attribute.
| Landmark | Element | aria-label needed? | Notes |
|---|---|---|---|
| Banner | <header> | No | Top-level only; one per page |
| Navigation | <nav> | Yes (if multiple exist) | Label each distinctly |
| Main | <main> | No | One per page; add id="main-content" tabIndex={-1} |
| Complementary | <aside> | Yes | Describe the supporting content |
| Contentinfo | <footer> | No | Top-level only; one per page |
| Search | <search> | No | Preferred over role="search" |
Rules: one <main> per page · label every <nav> when multiple exist · never nest <main> inside another landmark · give <main> id="main-content" tabIndex={-1} for skip-link targeting.
See semantic-html.md for the full AppLayout implementation.
Skip links bypass repeated navigation — required by WCAG 2.4.1 (Level A). Place <SkipLink /> as the first element in <App />. It uses visually-hidden (shown on focus via CSS) so it appears only on keyboard Tab.
Requirements:
#main-content) must have tabIndex={-1} to receive programmatic focusSee focus-management.md for the SkipLink component and App.tsx placement.
WCAG 2.4.2 (Level A): every page needs a unique, descriptive document.title that updates on route change.
useDocumentTitle(pageTitle) in every route component. Format: "Page Name — Site Name".<RouteAnnouncer /> once in <App /> — announces the new title to screen readers on navigation without moving focus.See focus-management.md for both implementations.
See keyboard-navigation.md for:
useKeyPress hookArrow + Home/End/Ctrl+Home/Ctrl+End)Previous/Next/Pause patternsSee aria-attributes.md for:
See focus-management.md for:
useFocusTrap (DSAI) — focus trapping for modalsscroll-margin-top fix for sticky headersPlace live region containers in your layout so they persist between renders — the container must exist in the DOM before content is injected:
// Polite — screen reader finishes current speech before announcing
<div aria-live="polite" aria-atomic="true">
{statusMessage}
</div>
// Assertive — interrupts screen reader immediately (use sparingly)
<div role="alert" aria-live="assertive">
{errorMessage}
</div>
// Status updates (WCAG 4.1.3 A) — equivalent to aria-live="polite" + aria-atomic
<div role="status">
{`${resultCount} results found`}
</div>
For imperative announcements (filter applied, item saved, search complete) without moving focus, use the useAnnounce hook from @/hooks/useAnnounce.
See aria-attributes.md for the implementation.
See reduced-motion.md for:
useReducedMotion hook@media (prefers-reduced-motion: reduce) (preferred approach)useReducedMotion integration@media (prefers-reduced-motion: reduce) global overrides| Content | Required Ratio |
|---|---|
| Normal text (< 18pt / < 14pt bold) | 4.5:1 |
| Large text (≥ 18pt or ≥ 14pt bold) | 3:1 |
| UI components & graphical objects (borders, icons, chart lines) | 3:1 (SC 1.4.11) |
| Inactive/disabled UI, pure decoration | No requirement |
(AAA) Enhanced contrast (SC 1.4.6): Normal text 7:1 · Large text 4.5:1
Use DSAI semantic color tokens — they map to CSS custom properties via the token pipeline:
| Pair | Token / Classes | Notes |
|---|---|---|
| Body text | text-body / bg-body | Uses var(--dsai-color-body) — passes ✔ |
| Primary action | bg-primary text-white | Uses var(--dsai-color-primary) — passes ✔ |
| Danger / error | bg-danger text-white | Uses var(--dsai-color-danger) — passes ✔ |
| Secondary surface | bg-secondary text-white | Uses var(--dsai-color-secondary) — passes ✔ |
| Muted text | text-body-secondary | ⚠ Verify at your font size |
Use WebAIM Contrast Checker or APCA Calculator to verify contrast for custom color combinations. DSAI tokens are pre-validated for AA compliance.
Input borders, button outlines, checkbox frames, focus rings, and icon-only controls must have 3:1 contrast against adjacent colors. Use DSAI tokens var(--dsai-border-color) / var(--dsai-color-primary) and verify in the contrast checker when customizing.
Use CSS :focus-visible pseudo-class (keyboard only — does not appear on mouse click). DSAI provides focus styles via CSS custom properties:
:focus-visible {
outline: 3px solid var(--dsai-color-primary);
outline-offset: 2px;
}
Add scroll-margin-top to prevent the focused element hiding behind a sticky header (WCAG 2.4.11 AA).
:focus alone for ring styles — it triggers on mouse toovar(--dsai-color-primary) for focus ring color (consistent across themes)See focus-management.md for the sticky-header fix.
// Icon-only button — always provide visually-hidden label
<button aria-label="Close modal">
<XIcon aria-hidden="true" />
</button>
// Or use visually-hidden span (Bootstrap class, either approach is valid)
<button>
<XIcon aria-hidden="true" />
<span className="visually-hidden">Close modal</span>
</button>
See forms-a11y.md for:
autocomplete attributes and WCAG 1.3.5 Identify Input Purpose (AA)fieldset/legend for grouped inputs (radio groups, checkboxes, date parts)See semantic-html.md for:
img alt text decision tree (informative vs decorative vs functional)alt="" + aria-hidden="true"role="img", <title>, <desc>, aria-labelledby<caption>, scope, headers attribute for complex tablesSee mobile-touch.md for:
pointerup, not pointerdown — WCAG 2.5.2See testing-a11y.md for:
@axe-core/playwright automated audits (recommended for CI)vitest-axe component-level unit testsUse this checklist for every new component or page before merge. Each item maps to a WCAG 2.2 success criterion.
<img> has meaningful alt; decorative images use alt="" and aria-hidden="true"autocomplete attribute valuesdocument.title that updates on route change<html lang="en"> (or appropriate BCP-47 language code)role="status" without moving focusThese exceed the required AA level. Implement where feasible to provide a superior experience.
| SC | Title | Implementation Guidance |
|---|---|---|
| 1.4.6 | Contrast Enhanced | Normal text 7:1 · Large text 4.5:1 |
| 2.3.3 | Animation from Interactions | Motion triggered by interaction can be disabled by the user — see reduced-motion.md |
| 2.4.12 | Focus Not Obscured (Enhanced) | Focused element not obscured at all — increase scroll-margin-top to fully clear sticky UI |
| 2.4.13 | Focus Appearance | Focus indicator ≥ 2px perimeter, 3:1 contrast between focused and unfocused states |
| 2.5.5 | Target Size Enhanced | Interactive targets ≥ 44×44 CSS pixels — use min-h-[44px] min-w-[44px] |
| 2.5.6 | Concurrent Input Mechanisms | Do not restrict the user to a single input modality (e.g., do not disable mouse when touch is detected) |
| 3.1.5 | Reading Level | If content exceeds lower secondary school reading level, provide supplemental simpler version |
| 3.3.9 | Accessible Auth Enhanced | No cognitive function test (including CAPTCHA with alternatives) in any auth step |
| Platform | Metadata | Shell commands | Packaging |
|---|---|---|---|
| Claude Code | YAML frontmatter in SKILL.md | ✅ Full !command`` support | n/a |
| Cowork | skills-rules.json (see skills-rules.json in this skill) | ✅ | .skill file via python -m scripts.package_skill skills/accessibility; use present_files to deliver packaged results |
| Claude.ai | Paste SKILL.md body into Project instructions; attach reference files as knowledge | ❌ No shell commands | n/a |
See skill-creator/reference/platform-specific.md for the full platform compatibility guide.
| Resource | URL |
|---|---|
| WCAG 2.2 Specification | https://www.w3.org/TR/WCAG22/ |
| WCAG 2.2 Quick Reference | https://www.w3.org/WAI/WCAG22/quickref/ |
| Understanding WCAG 2.2 | https://www.w3.org/WAI/WCAG22/Understanding/ |
| What's New in WCAG 2.2 | https://www.w3.org/WAI/standards-guidelines/wcag/new-in-22/ |
| ARIA Authoring Practices Guide (APG) | https://www.w3.org/WAI/ARIA/apg/ |
| WebAIM Contrast Checker | https://webaim.org/resources/contrastchecker/ |
| APCA Contrast Calculator | https://apcacontrast.com/ |
| axe DevTools Browser Extension | https://www.deque.com/axe/browser-extensions/ |
| WAVE Evaluation Tool | https://wave.webaim.org/ |
| NVDA Screen Reader (free, Windows) | https://www.nvaccess.org/ |
| Deque University — free courses | https://dequeuniversity.com/ |
| Inclusive Components — patterns | https://inclusive-components.design/ |