accessibility-testing
axe-coreとPlaywrightを用いて、ウェブサイトやアプリのアクセシビリティを検証し、WCAGに準拠しているか、キーボード操作やスクリーンリーダーでの利用に問題がないかなどを確認するSkill。
📜 元の英語説明(参考)
Accessibility testing with axe-core and Playwright. Use when checking WCAG compliance, finding a11y issues, ensuring keyboard navigation, or testing screen reader compatibility.
🇯🇵 日本人クリエイター向け解説
axe-coreとPlaywrightを用いて、ウェブサイトやアプリのアクセシビリティを検証し、WCAGに準拠しているか、キーボード操作やスクリーンリーダーでの利用に問題がないかなどを確認するSkill。
※ jpskill.com 編集部が日本のビジネス現場向けに補足した解説です。Skill本体の挙動とは独立した参考情報です。
下記のコマンドをコピーしてターミナル(Mac/Linux)または PowerShell(Windows)に貼り付けてください。 ダウンロード → 解凍 → 配置まで全自動。
mkdir -p ~/.claude/skills && cd ~/.claude/skills && curl -L -o accessibility-testing.zip https://jpskill.com/download/9347.zip && unzip -o accessibility-testing.zip && rm accessibility-testing.zip
$d = "$env:USERPROFILE\.claude\skills"; ni -Force -ItemType Directory $d | Out-Null; iwr https://jpskill.com/download/9347.zip -OutFile "$d\accessibility-testing.zip"; Expand-Archive "$d\accessibility-testing.zip" -DestinationPath $d -Force; ri "$d\accessibility-testing.zip"
完了後、Claude Code を再起動 → 普通に「動画プロンプト作って」のように話しかけるだけで自動発動します。
💾 手動でダウンロードしたい(コマンドが難しい人向け)
- 1. 下の青いボタンを押して
accessibility-testing.zipをダウンロード - 2. ZIPファイルをダブルクリックで解凍 →
accessibility-testingフォルダができる - 3. そのフォルダを
C:\Users\あなたの名前\.claude\skills\(Win)または~/.claude/skills/(Mac)へ移動 - 4. Claude Code を再起動
⚠️ ダウンロード・利用は自己責任でお願いします。当サイトは内容・動作・安全性について責任を負いません。
🎯 このSkillでできること
下記の説明文を読むと、このSkillがあなたに何をしてくれるかが分かります。Claudeにこの分野の依頼をすると、自動で発動します。
📦 インストール方法 (3ステップ)
- 1. 上の「ダウンロード」ボタンを押して .skill ファイルを取得
- 2. ファイル名の拡張子を .skill から .zip に変えて展開(macは自動展開可)
- 3. 展開してできたフォルダを、ホームフォルダの
.claude/skills/に置く- · macOS / Linux:
~/.claude/skills/ - · Windows:
%USERPROFILE%\.claude\skills\
- · macOS / Linux:
Claude Code を再起動すれば完了。「このSkillを使って…」と話しかけなくても、関連する依頼で自動的に呼び出されます。
詳しい使い方ガイドを見る →- 最終更新
- 2026-05-18
- 取得日時
- 2026-05-18
- 同梱ファイル
- 1
📖 Skill本文(日本語訳)
※ 原文(英語/中国語)を Gemini で日本語化したものです。Claude 自身は原文を読みます。誤訳がある場合は原文をご確認ください。
[Skill 名] accessibility-testing
axe-core を使用した WCAG 2.1 AA/AAA 準拠のための自動アクセシビリティテスト。Playwright と統合されています。
クイックスタート
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
test('homepage has no accessibility violations', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page }).analyze();
expect(results.violations).toEqual([]);
});
インストール
npm install -D @axe-core/playwright
基本的な使い方
ページ全体のスキャン
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
test('check entire page', async ({ page }) => {
await page.goto('/');
await page.waitForLoadState('networkidle');
const accessibilityScanResults = await new AxeBuilder({ page }).analyze();
expect(accessibilityScanResults.violations).toEqual([]);
});
特定の要素のスキャン
test('check navigation accessibility', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.include('nav')
.analyze();
expect(results.violations).toEqual([]);
});
動的なコンテンツの除外
test('check page excluding ads', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.exclude('.advertisement')
.exclude('#third-party-widget')
.analyze();
expect(results.violations).toEqual([]);
});
WCAG 準拠レベル
WCAG 2.1 レベル A
test('WCAG 2.1 Level A compliance', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag21a'])
.analyze();
expect(results.violations).toEqual([]);
});
WCAG 2.1 レベル AA (最も一般的な要件)
test('WCAG 2.1 Level AA compliance', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa', 'wcag21a', 'wcag21aa'])
.analyze();
expect(results.violations).toEqual([]);
});
WCAG 2.1 レベル AAA
test('WCAG 2.1 Level AAA compliance', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa', 'wcag2aaa', 'wcag21a', 'wcag21aa', 'wcag21aaa'])
.analyze();
expect(results.violations).toEqual([]);
});
一般的なルールカテゴリ
ベストプラクティスルール
test('accessibility best practices', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['best-practice'])
.analyze();
expect(results.violations).toEqual([]);
});
特定のルールのみ
test('check specific rules', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withRules(['color-contrast', 'image-alt', 'label', 'link-name'])
.analyze();
expect(results.violations).toEqual([]);
});
特定のルールを無効にする
test('check except known issues', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.disableRules(['color-contrast']) // Known issue, tracked separately
.analyze();
expect(results.violations).toEqual([]);
});
キーボードナビゲーションのテスト
タブ順序
test('verify tab order', async ({ page }) => {
await page.goto('/');
const expectedOrder = ['#search', '#nav-home', '#nav-about', '#nav-contact', '#main-content'];
for (const selector of expectedOrder) {
await page.keyboard.press('Tab');
const focused = await page.evaluate(() => document.activeElement?.id || document.activeElement?.className);
expect(`#${focused}`).toBe(selector);
}
});
フォーカスの可視性
test('focus indicators are visible', async ({ page }) => {
await page.goto('/');
await page.keyboard.press('Tab');
const focusedElement = page.locator(':focus');
const outline = await focusedElement.evaluate(el => {
const styles = window.getComputedStyle(el);
return styles.outline || styles.boxShadow;
});
expect(outline).not.toBe('none');
});
スキップリンク
test('skip link works', async ({ page }) => {
await page.goto('/');
// First tab should focus skip link
await page.keyboard.press('Tab');
await expect(page.locator(':focus')).toHaveText(/skip to/i);
// Enter should jump to main content
await page.keyboard.press('Enter');
await expect(page.locator(':focus')).toHaveAttribute('id', 'main-content');
});
色のコントラストのテスト
test('color contrast meets WCAG AA', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withRules(['color-contrast'])
.analyze();
if (results.violations.length > 0) {
console.log('Contrast violations:');
results.violations[0].nodes.forEach(node => {
console.log(` - ${node.html}`);
console.log(` ${node.failureSummary}`);
});
}
expect(results.violations).toEqual([]);
});
フォームのアクセシビリティ
test('form is accessible', async ({ page }) => {
await page.goto('/contact');
// Check labels
const inputs = page.locator('input:not([type="hidden"])');
const count = await inputs.count();
for (let i = 0; i < count; i++) {
const input = inputs.nth(i);
const id = await input.getAttribute('id');
const ariaLabel = await input.getAttribute('aria-label');
const ariaLabelledBy = await input.getAttribute('aria-labelledby');
const label = page.locator(`label[for="${id}"]`);
const hasLabel = await label.count() > 0 || ariaLabel || ariaLabelledBy;
expect(hasLabel).toBeTruthy();
}
// Run axe on form
const results = await new AxeBuilder({ page })
.include('form')
.analyze();
expect(results.violations).toEqual([]); 📜 原文 SKILL.md(Claudeが読む英語/中国語)を展開
Accessibility Testing with axe-core
Automated accessibility testing for WCAG 2.1 AA/AAA compliance using axe-core integrated with Playwright.
Quick Start
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
test('homepage has no accessibility violations', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page }).analyze();
expect(results.violations).toEqual([]);
});
Installation
npm install -D @axe-core/playwright
Basic Usage
Full Page Scan
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
test('check entire page', async ({ page }) => {
await page.goto('/');
await page.waitForLoadState('networkidle');
const accessibilityScanResults = await new AxeBuilder({ page }).analyze();
expect(accessibilityScanResults.violations).toEqual([]);
});
Specific Element Scan
test('check navigation accessibility', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.include('nav')
.analyze();
expect(results.violations).toEqual([]);
});
Exclude Dynamic Content
test('check page excluding ads', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.exclude('.advertisement')
.exclude('#third-party-widget')
.analyze();
expect(results.violations).toEqual([]);
});
WCAG Compliance Levels
WCAG 2.1 Level A
test('WCAG 2.1 Level A compliance', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag21a'])
.analyze();
expect(results.violations).toEqual([]);
});
WCAG 2.1 Level AA (Most Common Requirement)
test('WCAG 2.1 Level AA compliance', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa', 'wcag21a', 'wcag21aa'])
.analyze();
expect(results.violations).toEqual([]);
});
WCAG 2.1 Level AAA
test('WCAG 2.1 Level AAA compliance', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['wcag2a', 'wcag2aa', 'wcag2aaa', 'wcag21a', 'wcag21aa', 'wcag21aaa'])
.analyze();
expect(results.violations).toEqual([]);
});
Common Rule Categories
Best Practice Rules
test('accessibility best practices', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['best-practice'])
.analyze();
expect(results.violations).toEqual([]);
});
Specific Rules Only
test('check specific rules', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withRules(['color-contrast', 'image-alt', 'label', 'link-name'])
.analyze();
expect(results.violations).toEqual([]);
});
Disable Specific Rules
test('check except known issues', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.disableRules(['color-contrast']) // Known issue, tracked separately
.analyze();
expect(results.violations).toEqual([]);
});
Keyboard Navigation Testing
Tab Order
test('verify tab order', async ({ page }) => {
await page.goto('/');
const expectedOrder = ['#search', '#nav-home', '#nav-about', '#nav-contact', '#main-content'];
for (const selector of expectedOrder) {
await page.keyboard.press('Tab');
const focused = await page.evaluate(() => document.activeElement?.id || document.activeElement?.className);
expect(`#${focused}`).toBe(selector);
}
});
Focus Visibility
test('focus indicators are visible', async ({ page }) => {
await page.goto('/');
await page.keyboard.press('Tab');
const focusedElement = page.locator(':focus');
const outline = await focusedElement.evaluate(el => {
const styles = window.getComputedStyle(el);
return styles.outline || styles.boxShadow;
});
expect(outline).not.toBe('none');
});
Skip Links
test('skip link works', async ({ page }) => {
await page.goto('/');
// First tab should focus skip link
await page.keyboard.press('Tab');
await expect(page.locator(':focus')).toHaveText(/skip to/i);
// Enter should jump to main content
await page.keyboard.press('Enter');
await expect(page.locator(':focus')).toHaveAttribute('id', 'main-content');
});
Color Contrast Testing
test('color contrast meets WCAG AA', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withRules(['color-contrast'])
.analyze();
if (results.violations.length > 0) {
console.log('Contrast violations:');
results.violations[0].nodes.forEach(node => {
console.log(` - ${node.html}`);
console.log(` ${node.failureSummary}`);
});
}
expect(results.violations).toEqual([]);
});
Form Accessibility
test('form is accessible', async ({ page }) => {
await page.goto('/contact');
// Check labels
const inputs = page.locator('input:not([type="hidden"])');
const count = await inputs.count();
for (let i = 0; i < count; i++) {
const input = inputs.nth(i);
const id = await input.getAttribute('id');
const ariaLabel = await input.getAttribute('aria-label');
const ariaLabelledBy = await input.getAttribute('aria-labelledby');
const label = page.locator(`label[for="${id}"]`);
const hasLabel = await label.count() > 0 || ariaLabel || ariaLabelledBy;
expect(hasLabel).toBeTruthy();
}
// Run axe on form
const results = await new AxeBuilder({ page })
.include('form')
.analyze();
expect(results.violations).toEqual([]);
});
Image Accessibility
test('all images have alt text', async ({ page }) => {
await page.goto('/');
const images = page.locator('img');
const count = await images.count();
for (let i = 0; i < count; i++) {
const img = images.nth(i);
const alt = await img.getAttribute('alt');
const role = await img.getAttribute('role');
// Images must have alt OR be decorative (role="presentation")
const isAccessible = alt !== null || role === 'presentation' || role === 'none';
expect(isAccessible).toBeTruthy();
}
});
ARIA Testing
test('ARIA attributes are valid', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page })
.withTags(['cat.aria'])
.analyze();
expect(results.violations).toEqual([]);
});
Reporting
Detailed Violation Report
import { test, expect } from '@playwright/test';
import AxeBuilder from '@axe-core/playwright';
test('accessibility audit', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page }).analyze();
// Generate detailed report
if (results.violations.length > 0) {
console.log('\n=== Accessibility Violations ===\n');
results.violations.forEach(violation => {
console.log(`Rule: ${violation.id}`);
console.log(`Impact: ${violation.impact}`);
console.log(`Description: ${violation.description}`);
console.log(`Help: ${violation.helpUrl}`);
console.log(`Affected elements:`);
violation.nodes.forEach(node => {
console.log(` - ${node.html}`);
console.log(` ${node.failureSummary}`);
});
console.log('');
});
}
expect(results.violations).toEqual([]);
});
Save Report to File
import fs from 'fs';
test('save accessibility report', async ({ page }) => {
await page.goto('/');
const results = await new AxeBuilder({ page }).analyze();
// Save JSON report
fs.writeFileSync(
'accessibility-report.json',
JSON.stringify(results, null, 2)
);
// Save HTML report
const htmlReport = generateHtmlReport(results);
fs.writeFileSync('accessibility-report.html', htmlReport);
});
function generateHtmlReport(results: any): string {
return `
<!DOCTYPE html>
<html>
<head><title>Accessibility Report</title></head>
<body>
<h1>Accessibility Report</h1>
<p>Violations: ${results.violations.length}</p>
<p>Passes: ${results.passes.length}</p>
${results.violations.map(v => `
<div style="border:1px solid red;padding:10px;margin:10px 0">
<h3>${v.id}</h3>
<p><strong>Impact:</strong> ${v.impact}</p>
<p>${v.description}</p>
<p><a href="${v.helpUrl}">More info</a></p>
</div>
`).join('')}
</body>
</html>
`;
}
CI Integration
GitHub Actions
- name: Run accessibility tests
run: npx playwright test --grep @a11y
- name: Upload a11y report
if: failure()
uses: actions/upload-artifact@v4
with:
name: accessibility-report
path: accessibility-report.html
Best Practices
- Test early and often - Include a11y tests in CI
- Start with WCAG 2.1 AA - Most common legal requirement
- Test with real users - Automated tests catch ~30% of issues
- Test keyboard navigation - Essential for motor disabilities
- Test with screen readers - NVDA (Windows), VoiceOver (Mac)
- Fix critical issues first - Impact: critical > serious > moderate > minor
References
references/wcag-checklist.md- WCAG 2.1 compliance checklistreferences/common-issues.md- Most common a11y issues and fixes