Help us improve
Share bugs, ideas, or general feedback.
From clarc
Provides screen reader testing checklists (NVDA/VoiceOver/TalkBack), automated a11y tests (axe-core+Playwright, jest-axe, @axe-core/react), and accessible component patterns (modal, table, form, data viz).
npx claudepluginhub marvinrichter/clarc --plugin clarcHow this skill is triggered — by the user, by Claude, or both
Slash command
/clarc:accessibility-patterns-advancedThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
This skill extends `accessibility-patterns` with screen reader testing, automated testing, and accessible component patterns. Load `accessibility-patterns` first.
Automates accessibility testing for React components with axe-core and jest-axe, full-page Playwright scans, eslint-plugin-jsx-a11y, and CI integration to catch regressions.
Automates WCAG 2.1 AA accessibility checks using Playwright and axe-core: keyboard navigation, focus management, ARIA validation, and color contrast tests.
Implements WCAG 2.1/2.2 compliance, ARIA patterns, keyboard navigation, focus management, and accessibility testing for web components.
Share bugs, ideas, or general feedback.
This skill extends accessibility-patterns with screen reader testing, automated testing, and accessible component patterns. Load accessibility-patterns first.
Navigation shortcuts:
H / Shift+H — next/previous heading
F / Shift+F — next/previous form field
B / Shift+B — next/previous button
L / Shift+L — next/previous list
D / Shift+D — next/previous landmark
Tab — next interactive element
Enter/Space — activate button or link
Test checklist:
□ Turn on NVDA (Insert+Escape to toggle speech)
□ Navigate headings with H — is structure logical?
□ Tab through all interactive elements — are labels announced?
□ Submit a form with errors — are errors announced?
□ Open a modal — does focus move inside?
□ Close modal — does focus return to trigger?
Navigation shortcuts (VO = Control+Option):
VO+U — Rotor (shows headings, links, landmarks)
VO+Right/Left — next/previous element
VO+Space — activate element
VO+Command+H — next heading
Tab — next interactive element
Test checklist:
□ Open Rotor (VO+U) — are headings and landmarks correct?
□ Navigate to form — are labels associated?
□ Check dynamic content — does VoiceOver announce changes?
□ Test custom widgets — are roles and states announced?
// playwright.config.ts
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
test.describe('Accessibility', () => {
test('homepage has no critical violations', async ({ page }) => {
await page.goto('/');
await page.waitForLoadState('networkidle');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa', 'wcag21aa', 'wcag22aa'])
.exclude('.third-party-widget') // Exclude known third-party
.analyze();
// Filter to only serious/critical
const criticalViolations = results.violations.filter(
v => v.impact === 'critical' || v.impact === 'serious'
);
expect(criticalViolations).toEqual([]);
});
test('modal dialog is accessible', async ({ page }) => {
await page.goto('/dashboard');
await page.click('[data-testid="open-modal"]');
await page.waitForSelector('[role="dialog"]');
const results = await new AxeBuilder({ page })
.include('[role="dialog"]')
.analyze();
expect(results.violations).toEqual([]);
});
});
import { render } from '@testing-library/react';
import { axe, toHaveNoViolations } from 'jest-axe';
expect.extend(toHaveNoViolations);
describe('Button component', () => {
it('is accessible', async () => {
const { container } = render(
<Button onClick={() => {}}>Save changes</Button>
);
const results = await axe(container);
expect(results).toHaveNoViolations();
});
it('icon button has accessible label', async () => {
const { container } = render(
<IconButton aria-label="Close dialog" icon={<CloseIcon />} />
);
const results = await axe(container);
expect(results).toHaveNoViolations();
});
});
// index.tsx — dev-mode only
if (process.env.NODE_ENV !== 'production') {
import('@axe-core/react').then(({ default: axe }) => {
axe(React, ReactDOM, 1000); // Check every 1s, log to console
});
}
Automation finds ~30% of real issues. Always supplement with manual keyboard and screen reader testing.
interface ModalProps {
isOpen: boolean;
onClose: () => void;
title: string;
children: ReactNode;
}
function Modal({ isOpen, onClose, title, children }: ModalProps) {
const containerRef = useRef<HTMLDivElement>(null);
useFocusTrap(isOpen, containerRef);
if (!isOpen) return null;
return (
<div
role="dialog"
aria-modal="true"
aria-labelledby="modal-title"
ref={containerRef}
className="modal"
>
<h2 id="modal-title">{title}</h2>
{children}
<button onClick={onClose} aria-label={`Close ${title}`}>
✕
</button>
</div>
);
}
<table>
<caption>Monthly sales by region</caption>
<thead>
<tr>
<th scope="col">Region</th>
<th scope="col">Q1</th>
<th scope="col">Q2</th>
</tr>
</thead>
<tbody>
<tr>
<th scope="row">North America</th>
<td>$1.2M</td>
<td>$1.5M</td>
</tr>
</tbody>
</table>
<form novalidate>
<div class="form-field">
<label for="email">Email address <span aria-hidden="true">*</span></label>
<input
id="email"
type="email"
autocomplete="email"
aria-required="true"
aria-invalid="true"
aria-describedby="email-error"
/>
<p id="email-error" role="alert" class="error">
Please enter a valid email address.
</p>
</div>
</form>
// Every chart needs a text alternative
function SalesChart({ data }: { data: SalesData[] }) {
return (
<figure>
<figcaption>Monthly sales 2026 — peaks in Q2 and Q4</figcaption>
{/* Visual chart for sighted users */}
<canvas aria-hidden="true" ref={chartRef} />
{/* Text table for screen reader users */}
<table className="sr-only">
<caption>Monthly sales data 2026</caption>
<thead>
<tr><th>Month</th><th>Sales</th></tr>
</thead>
<tbody>
{data.map(d => (
<tr key={d.month}>
<td>{d.month}</td>
<td>{formatCurrency(d.sales)}</td>
</tr>
))}
</tbody>
</table>
</figure>
);
}
accessibility-patterns — WCAG criteria, ARIA patterns, keyboard navigation, focus trap, skip linksstorybook-patterns — addon-a11y for story-level accessibility checkse2e-testing — Playwright E2E tests (add axe checks to critical flows)