form-accessibility
π―Skillfrom bbeierle12/skill-mcp-claude
form-accessibility skill from bbeierle12/skill-mcp-claude
Part of
bbeierle12/skill-mcp-claude(74 items)
Installation
npm install @axe-core/reactSkill Details
WCAG 2.2 AA compliance for forms, ARIA patterns, focus management, keyboard navigation, and screen reader support. Use when implementing accessible forms in any framework. The compliance foundation that ensures forms work for everyone.
Overview
# Form Accessibility
WCAG 2.2 AA compliance patterns for forms. Ensures forms work for keyboard users, screen reader users, and users with cognitive or motor disabilities.
Quick Start
```tsx
// Accessible form field pattern
{/ 1. Visible label (never placeholder-only) /} Email * {/ 2. Hint text (separate from label) /} We'll send your confirmation here {/ 3. Input with full ARIA binding /} id="email" type="email" autoComplete="email" aria-required="true" aria-invalid={hasError} aria-describedby={hasError ? "email-error email-hint" : "email-hint"} /> {/ 4. Error message (announced by screen readers) /} {hasError && ( Please enter a valid email address )}
```
WCAG 2.2 Form Requirements
Critical Criteria
| Criterion | Level | Requirement | Implementation |
|-----------|-------|-------------|----------------|
| 1.3.1 Info & Relationships | A | Structure conveyed programmatically | , , aria-describedby |
| 1.3.5 Identify Input Purpose | AA | Input purpose identifiable | autocomplete attributes |
| 2.1.1 Keyboard | A | All functionality via keyboard | Tab order, focus management |
| 2.4.6 Headings & Labels | AA | Labels describe purpose | Descriptive, visible labels |
| 2.4.11 Focus Not Obscured | AA | Focus not hidden by other content | Scroll behavior, sticky elements |
| 2.5.8 Target Size | AA | 24Γ24px minimum touch target | Button/input sizing |
| 3.3.1 Error Identification | A | Errors identified and described | aria-invalid, error messages |
| 3.3.2 Labels or Instructions | A | Labels provided | Visible labels, not just placeholders |
| 3.3.3 Error Suggestion | AA | Suggestions for fixing errors | Actionable error messages |
| 3.3.7 Redundant Entry | A | Don't re-ask for info already provided | Form state management |
| 3.3.8 Accessible Authentication | AA | No cognitive function tests | No CAPTCHAs requiring text recognition |
New in WCAG 2.2 (October 2023)
2.4.11 Focus Not Obscured (AA)
```css
/ Ensure focus is never hidden by sticky headers /
.sticky-header {
position: sticky;
top: 0;
}
input:focus {
/ Browser should scroll input into view above sticky elements /
scroll-margin-top: 80px; / Height of sticky header /
}
```
2.5.8 Target Size (AA)
```css
/ Minimum 24Γ24px touch targets /
button,
input[type="submit"],
input[type="checkbox"],
input[type="radio"] {
min-width: 24px;
min-height: 24px;
}
/ Better: 44Γ44px for comfortable touch /
.touch-friendly {
min-width: 44px;
min-height: 44px;
}
```
3.3.7 Redundant Entry (A)
```tsx
// β BAD: Asking for email twice
// β GOOD: Ask once, show confirmation
Confirmation will be sent to: {email}
```
3.3.8 Accessible Authentication (AA)
```tsx
// β BAD: CAPTCHA requiring text recognition

// β GOOD: Alternative verification methods
Send verification code to email
```
ARIA Patterns
Error Message Binding
```tsx
// Pattern: aria-describedby links input to error
id="email"
aria-invalid={hasError ? "true" : "false"}
aria-describedby={hasError ? "email-error" : undefined}
/>
{hasError && (
{errorMessage}
)}
```
Multiple Descriptions
```tsx
// Pattern: Combine hint + error in aria-describedby
id="password"
aria-describedby={[
"password-hint",
hasError && "password-error"
].filter(Boolean).join(" ")}
/>
Must be at least 8 characters
{hasError && {error}}
```
Required Fields
```tsx
// Pattern: Announce required status
Name
*
{/ Visual indicator hidden from SR, aria-required announces it /}
id="name"
aria-required="true"
/>
// Alternative: Required in label (simpler)
```
Field Groups
```tsx
// Pattern: fieldset + legend for related fields
```
Radio/Checkbox Groups
```tsx
// Pattern: fieldset groups options, legend is the question
Phone
```
Focus Management
Focus on First Error
```tsx
// On form submit with errors, focus first invalid field
function handleSubmit(e: FormEvent) {
e.preventDefault();
const firstError = formRef.current?.querySelector('[aria-invalid="true"]');
if (firstError) {
(firstError as HTMLElement).focus();
return;
}
// Submit if valid
submitForm();
}
```
Focus on Step Change (Multi-step)
```tsx
// Move focus to step heading when changing steps
function goToStep(stepNumber: number) {
setCurrentStep(stepNumber);
// Wait for render, then focus
requestAnimationFrame(() => {
const heading = document.getElementById(step-${stepNumber}-heading);
heading?.focus();
});
}
// Heading must be focusable
Shipping Address
```
Skip Links
```tsx
// Allow skipping to form
Skip to form
{/ Form content /}
// CSS for skip link
.skip-link {
position: absolute;
top: -40px;
left: 0;
z-index: 100;
}
.skip-link:focus {
top: 0;
}
```
Focus Trap (Modals)
```tsx
// Keep focus within modal form
function FocusTrap({ children }) {
const trapRef = useRef
useEffect(() => {
const trap = trapRef.current;
if (!trap) return;
const focusableElements = trap.querySelectorAll(
'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
);
const firstElement = focusableElements[0] as HTMLElement;
const lastElement = focusableElements[focusableElements.length - 1] as HTMLElement;
function handleKeyDown(e: KeyboardEvent) {
if (e.key !== 'Tab') return;
if (e.shiftKey && document.activeElement === firstElement) {
e.preventDefault();
lastElement.focus();
} else if (!e.shiftKey && document.activeElement === lastElement) {
e.preventDefault();
firstElement.focus();
}
}
trap.addEventListener('keydown', handleKeyDown);
firstElement?.focus();
return () => trap.removeEventListener('keydown', handleKeyDown);
}, []);
return
}
```
Color & Contrast
Error States (Colorblind-Safe)
```css
/ β BAD: Color only /
.error {
border-color: red;
}
/ β GOOD: Color + icon + text /
.field-error {
border-color: #dc2626;
border-width: 2px;
}
.field-error::after {
content: "";
background-image: url("data:image/svg+xml,..."); / Error icon /
}
.error-message {
color: #dc2626;
font-weight: 500;
}
.error-message::before {
content: "β "; / Text indicator /
}
```
Focus Indicators
```css
/ Focus must have 3:1 contrast ratio /
input:focus {
outline: 2px solid #2563eb;
outline-offset: 2px;
}
/ For dark backgrounds /
input:focus {
outline: 2px solid #60a5fa;
outline-offset: 2px;
}
/ Never remove outline without replacement /
/ β BAD /
input:focus {
outline: none;
}
/ β GOOD: Custom focus style /
input:focus {
outline: none;
box-shadow: 0 0 0 3px rgba(37, 99, 235, 0.5);
}
```
Validation States (Colorblind-Friendly)
```tsx
// Use icons + text, not just color
function ValidationIndicator({ state }: { state: 'valid' | 'invalid' | 'idle' }) {
if (state === 'idle') return null;
return (
indicator ${state}} aria-hidden="true">
{state === 'valid' && 'β'}
{state === 'invalid' && 'β'}
);
}
```
Keyboard Navigation
Tab Order
```tsx
// Natural tab order (no positive tabindex needed)
// β BAD: Manual tab order
// β GOOD: Natural DOM order
{/ tabIndex implicitly 0 /}
```
Escape Key Handling
```tsx
// Allow Escape to close dropdowns, cancel modals
function Modal({ onClose, children }) {
useEffect(() => {
function handleEscape(e: KeyboardEvent) {
if (e.key === 'Escape') {
onClose();
}
}
document.addEventListener('keydown', handleEscape);
return () => document.removeEventListener('keydown', handleEscape);
}, [onClose]);
return
}
```
Enter to Submit
```tsx
// Forms submit on Enter by default
// For buttons that shouldn't submit:
Add Item
// For preventing Enter submit on specific fields:
onKeyDown={(e) => {
if (e.key === 'Enter') {
e.preventDefault();
// Do something else
}
}}
/>
```
Live Regions
Error Announcements
```tsx
// Announce errors when they appear
{errorCount > 0 && ${errorCount} errors in form}
// Or use role="alert" for immediate announcement
{hasError && (
{errorMessage}
)}
```
Loading States
```tsx
// Announce loading state
{isLoading ? (
<>
Loading...
Submitting form, please wait
>
) : (
'Submit'
)}
// Or use aria-busy
{/ ... /}
```
Success Messages
```tsx
// Announce successful submission
{isSuccess && (
Form submitted successfully!
)}
```
Screen Reader Only Content
```css
/ Visually hidden but announced by screen readers /
.sr-only {
position: absolute;
width: 1px;
height: 1px;
padding: 0;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
white-space: nowrap;
border: 0;
}
/ Allow focus for skip links /
.sr-only-focusable:focus {
position: static;
width: auto;
height: auto;
overflow: visible;
clip: auto;
white-space: normal;
}
```
Testing Accessibility
Automated Tools
```bash
# axe-core (recommended)
npm install @axe-core/react
# In development
import React from 'react';
import ReactDOM from 'react-dom';
import axe from '@axe-core/react';
if (process.env.NODE_ENV !== 'production') {
axe(React, ReactDOM, 1000);
}
```
Manual Testing Checklist
- Keyboard only: Can you complete the form using only Tab, Enter, Space, and Arrow keys?
- Screen reader: Does VoiceOver/NVDA announce labels, errors, and required status?
- Zoom 200%: Is the form usable at 200% browser zoom?
- High contrast: Is everything visible in Windows High Contrast mode?
- Focus visible: Can you always see which element is focused?
Testing Script
```typescript
// Automated accessibility test
import { axe, toHaveNoViolations } from 'jest-axe';
expect.extend(toHaveNoViolations);
test('form is accessible', async () => {
const { container } = render(
const results = await axe(container);
expect(results).toHaveNoViolations();
});
test('error state is accessible', async () => {
const { container } = render(
// Trigger error
fireEvent.blur(screen.getByLabelText(/email/i));
const results = await axe(container);
expect(results).toHaveNoViolations();
});
```
File Structure
```
form-accessibility/
βββ SKILL.md
βββ references/
β βββ wcag-2.2-forms.md # Full WCAG criteria breakdown
β βββ aria-patterns.md # Complete ARIA reference
βββ scripts/
βββ aria-form-wrapper.tsx # Automatic ARIA binding
βββ focus-manager.ts # Focus trap, error focus
βββ error-announcer.ts # Live region management
βββ accessibility-validator.ts # Runtime a11y checks
```
Reference
references/wcag-2.2-forms.mdβ Complete WCAG 2.2 criteria for formsreferences/aria-patterns.mdβ Detailed ARIA implementation patterns
More from this repository10
gsap-react skill from bbeierle12/skill-mcp-claude
Enables advanced scroll-based animations and interactions using GSAP ScrollTrigger, allowing precise control of element animations triggered by scrolling.
gsap-fundamentals skill from bbeierle12/skill-mcp-claude
gsap-sequencing skill from bbeierle12/skill-mcp-claude
shader-effects skill from bbeierle12/skill-mcp-claude
gsap-router skill from bbeierle12/skill-mcp-claude
d3js-visualization skill from bbeierle12/skill-mcp-claude
particles-gpu skill from bbeierle12/skill-mcp-claude
shader-sdf skill from bbeierle12/skill-mcp-claude
Generates a dynamic, interactive React form with real-time validation and state management for capturing structured user input.