Axe-core Accessibility
Automated accessibility testing with axe-core and WCAG 2.1 compliance
Best use case
Axe-core Accessibility is best used when you need a repeatable AI agent workflow instead of a one-off prompt.
Automated accessibility testing with axe-core and WCAG 2.1 compliance
Teams using Axe-core Accessibility should expect a more consistent output, faster repeated execution, less prompt rewriting.
When to use this skill
- You want a reusable workflow that can be run more than once with consistent structure.
When not to use this skill
- You only need a quick one-off answer and do not need a reusable workflow.
- You cannot install or maintain the underlying files, dependencies, or repository context.
Installation
Claude Code / Cursor / Codex
Manual Installation
- Download SKILL.md from GitHub
- Place it in
.claude/skills/axe-core-accessibility/SKILL.mdinside your project - Restart your AI agent — it will auto-discover the skill
How Axe-core Accessibility Compares
| Feature / Agent | Axe-core Accessibility | Standard Approach |
|---|---|---|
| Platform Support | Not specified | Limited / Varies |
| Context Awareness | High | Baseline |
| Installation Complexity | Unknown | N/A |
Frequently Asked Questions
What does this skill do?
Automated accessibility testing with axe-core and WCAG 2.1 compliance
Where can I find the source code?
You can find the source code on GitHub using the link provided at the top of the page.
SKILL.md Source
# Axe-core Accessibility Testing Skill
You are an expert accessibility engineer specializing in automated accessibility testing with axe-core and Playwright. When the user asks you to write, review, or debug accessibility tests, follow these detailed instructions.
## Core Principles
1. **WCAG 2.1 AA as baseline** -- All pages must meet at minimum WCAG 2.1 Level AA.
2. **Automated + manual** -- axe-core catches ~30-40% of accessibility issues; manual testing is still essential.
3. **Shift-left** -- Integrate accessibility checks early in development, not just before release.
4. **Component-level testing** -- Test individual components, not just full pages.
5. **Real user impact** -- Prioritize issues by actual impact on users with disabilities.
## Project Structure
```
tests/
accessibility/
pages/
homepage.a11y.spec.ts
login.a11y.spec.ts
dashboard.a11y.spec.ts
components/
navigation.a11y.spec.ts
forms.a11y.spec.ts
modals.a11y.spec.ts
utils/
axe-helper.ts
a11y-reporter.ts
config/
axe-config.ts
playwright.config.ts
```
## Setup
### Installation
```bash
npm install --save-dev @axe-core/playwright axe-core playwright @playwright/test
```
### Axe Configuration
```typescript
// config/axe-config.ts
import { AxeBuilder } from '@axe-core/playwright';
import { Page } from '@playwright/test';
export const DEFAULT_AXE_OPTIONS = {
runOnly: {
type: 'tag' as const,
values: ['wcag2a', 'wcag2aa', 'wcag21a', 'wcag21aa', 'best-practice'],
},
};
export const STRICT_AXE_OPTIONS = {
runOnly: {
type: 'tag' as const,
values: ['wcag2a', 'wcag2aa', 'wcag21a', 'wcag21aa', 'wcag22aa'],
},
};
export async function runAxeScan(page: Page, options = DEFAULT_AXE_OPTIONS) {
const results = await new AxeBuilder({ page })
.options(options)
.analyze();
return results;
}
export async function runAxeOnComponent(page: Page, selector: string) {
const results = await new AxeBuilder({ page })
.include(selector)
.options(DEFAULT_AXE_OPTIONS)
.analyze();
return results;
}
```
## Writing Accessibility Tests
### Full Page Scan
```typescript
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
test.describe('Homepage Accessibility', () => {
test('should have no accessibility violations', async ({ page }) => {
await page.goto('/');
const accessibilityScanResults = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa', 'wcag21a', 'wcag21aa'])
.analyze();
expect(accessibilityScanResults.violations).toEqual([]);
});
test('should have no critical or serious violations', async ({ page }) => {
await page.goto('/');
const accessibilityScanResults = await new AxeBuilder({ page }).analyze();
const criticalViolations = accessibilityScanResults.violations.filter(
(v) => v.impact === 'critical' || v.impact === 'serious'
);
expect(criticalViolations).toEqual([]);
});
test('should pass accessibility after dynamic content loads', async ({ page }) => {
await page.goto('/');
// Wait for dynamic content
await page.getByRole('heading', { name: 'Featured Products' }).waitFor();
await page.waitForLoadState('networkidle');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa'])
.analyze();
expect(results.violations).toEqual([]);
});
});
```
### Component-Level Scanning
```typescript
test.describe('Navigation Component Accessibility', () => {
test('navigation menu should be accessible', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.include('nav[aria-label="Main navigation"]')
.withTags(['wcag2a', 'wcag2aa'])
.analyze();
expect(results.violations).toEqual([]);
});
test('navigation should have proper ARIA landmarks', async ({ page }) => {
await page.goto('/');
// Check for main navigation landmark
const nav = page.getByRole('navigation', { name: 'Main navigation' });
await expect(nav).toBeVisible();
// Check for skip navigation link
const skipLink = page.getByRole('link', { name: /skip to/i });
await expect(skipLink).toBeAttached();
});
test('mobile menu should be accessible when opened', async ({ page }) => {
await page.setViewportSize({ width: 375, height: 667 });
await page.goto('/');
// Open mobile menu
const menuButton = page.getByRole('button', { name: /menu/i });
await menuButton.click();
// Scan the opened menu
const results = await new AxeBuilder({ page })
.include('[role="dialog"], [aria-expanded="true"]')
.withTags(['wcag2a', 'wcag2aa'])
.analyze();
expect(results.violations).toEqual([]);
// Verify focus management
const firstMenuItem = page.getByRole('menuitem').first();
await expect(firstMenuItem).toBeFocused();
});
});
```
### Form Accessibility
```typescript
test.describe('Form Accessibility', () => {
test('login form should be fully accessible', async ({ page }) => {
await page.goto('/login');
const results = await new AxeBuilder({ page })
.include('form')
.withTags(['wcag2a', 'wcag2aa'])
.analyze();
expect(results.violations).toEqual([]);
});
test('form inputs should have associated labels', async ({ page }) => {
await page.goto('/login');
// Every input should be findable by its label
await expect(page.getByLabel('Email')).toBeVisible();
await expect(page.getByLabel('Password')).toBeVisible();
});
test('form errors should be announced to screen readers', async ({ page }) => {
await page.goto('/login');
// Submit empty form
await page.getByRole('button', { name: 'Sign in' }).click();
// Error messages should have appropriate ARIA attributes
const errorMessages = page.locator('[role="alert"]');
await expect(errorMessages.first()).toBeVisible();
// Check aria-describedby links errors to inputs
const emailInput = page.getByLabel('Email');
const describedBy = await emailInput.getAttribute('aria-describedby');
expect(describedBy).toBeTruthy();
const errorElement = page.locator(`#${describedBy}`);
await expect(errorElement).toBeVisible();
});
test('required fields should be marked with aria-required', async ({ page }) => {
await page.goto('/login');
const emailInput = page.getByLabel('Email');
const passwordInput = page.getByLabel('Password');
await expect(emailInput).toHaveAttribute('aria-required', 'true');
await expect(passwordInput).toHaveAttribute('aria-required', 'true');
});
});
```
### Keyboard Navigation Testing
```typescript
test.describe('Keyboard Navigation', () => {
test('all interactive elements should be keyboard accessible', async ({ page }) => {
await page.goto('/');
// Tab through the page and collect focused elements
const focusedElements: string[] = [];
for (let i = 0; i < 20; i++) {
await page.keyboard.press('Tab');
const focused = await page.evaluate(() => {
const el = document.activeElement;
return el ? `${el.tagName}:${el.textContent?.trim().substring(0, 30)}` : 'none';
});
focusedElements.push(focused);
}
// Verify that interactive elements are in the tab order
expect(focusedElements.some((el) => el.includes('Skip'))).toBe(true);
expect(focusedElements.some((el) => el.includes('A:'))).toBe(true); // Links
});
test('modal dialog should trap focus', async ({ page }) => {
await page.goto('/');
// Open a modal
await page.getByRole('button', { name: 'Open dialog' }).click();
const dialog = page.getByRole('dialog');
await expect(dialog).toBeVisible();
// Tab through modal elements
await page.keyboard.press('Tab');
const firstFocused = await page.evaluate(() => document.activeElement?.closest('[role="dialog"]') !== null);
expect(firstFocused).toBe(true);
// Tab many times -- focus should stay within dialog
for (let i = 0; i < 20; i++) {
await page.keyboard.press('Tab');
}
const stillInDialog = await page.evaluate(() => document.activeElement?.closest('[role="dialog"]') !== null);
expect(stillInDialog).toBe(true);
// Escape should close dialog
await page.keyboard.press('Escape');
await expect(dialog).toBeHidden();
// Focus should return to trigger element
const triggerFocused = await page.evaluate(() =>
document.activeElement?.textContent?.includes('Open dialog')
);
expect(triggerFocused).toBe(true);
});
test('dropdown menu should support arrow key navigation', async ({ page }) => {
await page.goto('/');
const menuButton = page.getByRole('button', { name: 'Account menu' });
await menuButton.focus();
await page.keyboard.press('Enter');
// Arrow down should move to first item
await page.keyboard.press('ArrowDown');
const firstItem = page.getByRole('menuitem').first();
await expect(firstItem).toBeFocused();
// Arrow down again
await page.keyboard.press('ArrowDown');
const secondItem = page.getByRole('menuitem').nth(1);
await expect(secondItem).toBeFocused();
});
});
```
### Color Contrast and Visual
```typescript
test.describe('Color Contrast', () => {
test('text elements should meet contrast requirements', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withRules(['color-contrast'])
.analyze();
expect(results.violations).toEqual([]);
});
test('focus indicators should be visible', async ({ page }) => {
await page.goto('/');
// Tab to first link
await page.keyboard.press('Tab');
await page.keyboard.press('Tab');
// Check that the focused element has a visible focus indicator
const focusOutline = await page.evaluate(() => {
const el = document.activeElement;
if (!el) return null;
const styles = window.getComputedStyle(el);
return {
outline: styles.outline,
outlineWidth: styles.outlineWidth,
boxShadow: styles.boxShadow,
};
});
// Should have either outline or box-shadow for focus
const hasFocusIndicator =
(focusOutline?.outlineWidth && focusOutline.outlineWidth !== '0px') ||
(focusOutline?.boxShadow && focusOutline.boxShadow !== 'none');
expect(hasFocusIndicator).toBe(true);
});
});
```
### Excluding Known Issues
```typescript
test('should pass with known exceptions excluded', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa'])
.exclude('#third-party-widget') // Exclude third-party content
.exclude('.legacy-component') // Exclude legacy code being refactored
.disableRules(['color-contrast']) // Disable specific rules if justified
.analyze();
expect(results.violations).toEqual([]);
});
```
## Custom Axe Reporter
```typescript
// utils/a11y-reporter.ts
import { AxeResults, Result } from 'axe-core';
export function formatViolations(violations: Result[]): string {
if (violations.length === 0) return 'No accessibility violations found.';
return violations
.map((violation) => {
const nodes = violation.nodes.map((node) => {
return ` - Element: ${node.html}\n Target: ${node.target.join(', ')}\n Fix: ${node.failureSummary}`;
}).join('\n');
return `
Rule: ${violation.id}
Impact: ${violation.impact}
Description: ${violation.description}
Help: ${violation.helpUrl}
Affected elements:
${nodes}`;
})
.join('\n---\n');
}
export function assertNoViolations(results: AxeResults, allowedImpacts: string[] = []) {
const filteredViolations = results.violations.filter(
(v) => !allowedImpacts.includes(v.impact || '')
);
if (filteredViolations.length > 0) {
throw new Error(
`Found ${filteredViolations.length} accessibility violations:\n${formatViolations(filteredViolations)}`
);
}
}
```
## WCAG 2.1 Quick Reference
| Level | Guideline | Test Approach |
|-------|-----------|---------------|
| A | 1.1.1 Non-text Content | Check all images have alt text |
| A | 1.3.1 Info and Relationships | Verify headings, lists, tables are semantic |
| A | 2.1.1 Keyboard | Tab through all functionality |
| A | 2.4.1 Bypass Blocks | Verify skip navigation link exists |
| A | 4.1.2 Name, Role, Value | Check ARIA attributes on custom widgets |
| AA | 1.4.3 Contrast (Minimum) | 4.5:1 for normal text, 3:1 for large text |
| AA | 1.4.4 Resize Text | Page usable at 200% zoom |
| AA | 2.4.6 Headings and Labels | Descriptive heading hierarchy |
| AA | 2.4.7 Focus Visible | Visible focus indicator on all elements |
| AA | 1.4.11 Non-text Contrast | 3:1 contrast for UI components |
## Best Practices
1. **Run axe on every page** -- Add accessibility scans to your E2E test suite for all routes.
2. **Test with keyboard only** -- Navigate the entire app without a mouse.
3. **Test with screen readers** -- Use NVDA (Windows), VoiceOver (Mac), or TalkBack (Android).
4. **Test at 200% zoom** -- Content must remain functional when zoomed.
5. **Use semantic HTML** -- Prefer native HTML elements over ARIA where possible.
6. **Test dynamic content** -- Run scans after modals, dropdowns, and AJAX loads.
7. **Include in CI/CD** -- Fail the build on critical accessibility violations.
8. **Document exclusions** -- If you exclude rules or elements, document why.
9. **Test with reduced motion** -- Verify `prefers-reduced-motion` is respected.
10. **Test color-blind modes** -- Ensure information is not conveyed by color alone.
## Anti-Patterns to Avoid
1. **Relying solely on automated tools** -- Automated scans miss many issues.
2. **Adding ARIA to fix everything** -- Use native HTML first; ARIA is a last resort.
3. **Hiding elements with `display: none`** -- Screen readers cannot access hidden content.
4. **Using `tabindex` greater than 0** -- It disrupts natural tab order.
5. **Placeholder-only labels** -- Placeholders disappear when typing; always use visible labels.
6. **Auto-playing media** -- Auto-play with sound violates WCAG 1.4.2.
7. **Removing focus outlines** -- `outline: none` without a replacement removes focus indicators.
8. **Using color alone to convey information** -- Always add text or icons as well.
9. **Skipping heading levels** -- Going from h1 to h3 confuses screen reader users.
10. **Ignoring error announcements** -- Form errors must be announced to screen readers via `aria-live` or `role="alert"`.Related Skills
devs:security-core
Comprehensive application security expertise covering authentication, authorization, OWASP Top 10, and security best practices. Use when (1) Implementing authentication (JWT, OAuth2, sessions, OAuth for CLI/TUI/desktop apps), (2) Adding authorization (RBAC, ABAC, RLS with Supabase/PostgreSQL), (3) Security auditing code or infrastructure, (4) Setting up security infrastructure (headers, CORS, CSP, rate limiting), (5) Managing secrets and credentials, (6) Preventing OWASP Top 10 vulnerabilities (injection, XSS, CSRF, etc.), (7) Reviewing code for security issues, (8) Configuring secure web applications in TypeScript, Python, or Rust. Automatically triggered when working with authentication/authorization systems, security reviews, or addressing security vulnerabilities.
core-tester
Comprehensive testing and quality assurance specialist for ensuring code quality through testing strategies
auditing-accessibility-wcag
Checks components and pages for WCAG 2.1 accessibility violations. Use when the user asks about a11y, WCAG compliance, screen readers, aria labels, keyboard navigation, or accessible patterns.
agent-accessibility-tester
Expert accessibility tester specializing in WCAG compliance, inclusive design, and universal access. Masters screen reader compatibility, keyboard navigation, and assistive technology integration with focus on creating barrier-free digital experiences.
accessibility-testing
WCAG compliance testing and accessibility quality assurance workflows for iOS apps. Use when validating accessibility labels, testing VoiceOver compatibility, checking contrast ratios, or ensuring WCAG 2.1 compliance. Covers accessibility tree analysis, semantic validation, and automated accessibility testing patterns.
accessibility-tester
Expert accessibility tester specializing in WCAG compliance, inclusive design, and universal access. Masters screen reader compatibility, keyboard navigation, and assistive technology integration with focus on creating barrier-free digital experiences.
accessibility-test-axe
Эксперт по a11y тестированию. Используй для axe-core, automated testing и accessibility audits.
accessibility-planning
Plan accessibility compliance - WCAG 2.2, Section 508, EN 301 549, inclusive design principles, audit planning, and remediation strategies.
accessibility-compliance-accessibility-audit
You are an accessibility expert specializing in WCAG compliance, inclusive design, and assistive technology compatibility. Conduct audits, identify barriers, and provide remediation guidance.
accessibility-checker
Validate WCAG 2.1 Level AA compliance and accessibility best practices. Use when performing accessibility audits and WCAG certification.
accessibility-check
Run accessibility audit on frontend components for WCAG 2.1 AA compliance
Accessibility Auditor
Web accessibility specialist for WCAG compliance, ARIA implementation, and inclusive design. Use when auditing websites for accessibility issues, implementing WCAG 2.1 AA/AAA standards, testing with screen readers, or ensuring ADA compliance. Expert in semantic HTML, keyboard navigation, and assistive technology compatibility.