Validate UI components for WCAG 2.2 Level AA compliance. Use when checking accessibility, color contrast, keyboard navigation, screen reader support, or ARIA attributes.
View on GitHubfusengine/agents
fuse-design
plugins/design-expert/skills/validating-accessibility/SKILL.md
January 22, 2026
Select agents to install to:
npx add-skill https://github.com/fusengine/agents/blob/main/plugins/design-expert/skills/validating-accessibility/SKILL.md -a claude-code --skill validating-accessibilityInstallation paths:
.claude/skills/validating-accessibility/# Validating Accessibility
Ensure WCAG 2.2 Level AA compliance.
## Quick Checklist
```
[ ] Color contrast: 4.5:1 (text), 3:1 (UI/large text)
[ ] Keyboard: All interactive elements focusable
[ ] Focus: Visible focus indicators
[ ] ARIA: Correct attributes on custom controls
[ ] Motion: prefers-reduced-motion respected
[ ] Semantic: Proper heading hierarchy (h1→h2→h3)
[ ] Labels: All form inputs have labels
[ ] Alt text: All images have alt attributes
```
## Color Contrast
```
Normal text (<18px): 4.5:1 minimum
Large text (≥18px): 3:1 minimum
UI components: 3:1 minimum
```
```css
/* Safe high-contrast pairs */
--text-on-light: oklch(15% 0 0); /* ~16:1 on white */
--text-on-dark: oklch(95% 0 0); /* ~14:1 on black */
```
## Keyboard Navigation
```tsx
// Visible focus indicator (MANDATORY)
className="focus:outline-none focus-visible:ring-2 focus-visible:ring-primary"
// Skip link (first element in body)
<a href="#main" className="sr-only focus:not-sr-only">
Skip to main content
</a>
```
## Semantic HTML
```html
<!-- Correct heading hierarchy -->
<h1>Page Title</h1>
<h2>Section</h2>
<h3>Subsection</h3>
<!-- Landmarks -->
<header role="banner">
<nav role="navigation">
<main role="main" id="main">
<footer role="contentinfo">
```
## ARIA Patterns
```tsx
// Button with icon only
<button aria-label="Close dialog">
<X className="h-4 w-4" />
</button>
// Expandable content
<button aria-expanded={isOpen} aria-controls="panel">
Toggle
</button>
<div id="panel" aria-hidden={!isOpen}>Content</div>
// Form with description
<input aria-describedby="hint" />
<p id="hint">Must be 8+ characters</p>
```
## Motion Accessibility
```tsx
// Framer Motion with reduced motion
import { useReducedMotion } from "framer-motion";
function Component() {
const shouldReduce = useReducedMotion();
return (
<motion.div
animate={shouldReduce ? {} : { y: 0, opacity: 1 }}
/>
);
}
```
```css
/* CSS fallback */
@media (prefers-reduced-