accessibility
Web Accessibility
Accessibility is not a feature you bolt on at the end -- it is a quality attribute that must be considered from the first line of markup. Building accessible interfaces means that people with visual, motor, auditory, or cognitive disabilities can perceive, navigate, and interact with your application. It also means better usability for everyone: keyboard power users, people on slow connections, users with temporary injuries, and those in constrained environments.
Why Accessibility Matters
| Dimension | Impact |
|---|---|
| Legal | Legislation in most jurisdictions (ADA, EAA, Section 508, EN 301 549) requires digital products to be accessible. Non-compliance carries litigation risk and financial penalties. |
| Ethical | Roughly 16% of the global population lives with some form of disability. Excluding them from digital services is a choice, not an inevitability. |
| Business | Accessible products reach a wider audience, improve SEO (structured content helps crawlers), reduce support costs, and correlate with higher overall usability scores. |
| Technical quality | Accessibility constraints force clean markup, proper semantics, and separation of concerns -- all of which improve maintainability. |
WCAG Principles: POUR
The Web Content Accessibility Guidelines organize all success criteria under four principles. Every accessibility requirement maps to at least one.
| Principle | Question It Answers | Examples |
|---|---|---|
| Perceivable | Can users sense the content? | Text alternatives for images, captions for video, sufficient color contrast, resizable text |
| Operable | Can users interact with every control? | Keyboard operability, enough time to complete tasks, no seizure-triggering animations, clear navigation |
| Understandable | Can users comprehend the content and UI behavior? | Readable language, predictable navigation, input assistance and error messages |
| Robust | Does it work across assistive technologies? | Valid markup, proper use of ARIA, compatibility with screen readers and other tools |
Conformance Levels
WCAG defines three levels. Each higher level includes all criteria from the levels below it.
| Level | Target Audience | Typical Requirement |
|---|---|---|
| A | Bare minimum -- removes the most severe barriers | Most legal and procurement requirements start here |
| AA | Industry standard -- addresses the majority of barriers for most users | Required by ADA, EAA, Section 508, and most organizational policies |
| AAA | Highest standard -- not always achievable for all content types | Aspirational goal; apply selectively where feasible |
For most projects, target Level AA. It covers the vast majority of real-world accessibility needs without imposing requirements that conflict with certain content types.
Semantic HTML Fundamentals
Native HTML elements carry built-in semantics, keyboard behavior, and screen reader announcements. Using the right element is the single most effective accessibility technique.
| Instead Of | Use | Why |
|---|---|---|
<div onclick="..."> |
<button> |
Buttons are focusable, announce their role, and respond to Enter and Space |
<span class="link"> |
<a href="..."> |
Links announce as "link," support middle-click, and appear in link lists |
<div class="header"> |
<header>, <nav>, <main>, <footer> |
Landmark elements let screen reader users jump between page regions |
<div class="list"> |
<ul> / <ol> with <li> |
Lists announce item count and position ("item 3 of 7") |
<div class="table"> |
<table> with <th> |
Table headers associate data cells with their labels for screen readers |
Styled <div> for input |
<input>, <select>, <textarea> |
Native form controls have label association, validation, and assistive tech support built in |
Heading Hierarchy
Headings create an outline that screen reader users navigate like a table of contents. Follow these rules:
- One
<h1>per page that describes the page purpose - Never skip levels (do not jump from
<h2>to<h4>) - Use headings for structure, not for visual styling -- CSS handles appearance
Common Accessibility Issues
| Issue | Impact | Fix |
|---|---|---|
| Missing alt text on images | Screen readers announce the filename or nothing | Add descriptive alt; use alt="" for purely decorative images |
| Insufficient color contrast | Users with low vision cannot read text | Meet 4.5:1 ratio for normal text, 3:1 for large text (AA) |
| No keyboard access to interactive elements | Keyboard and switch users are completely blocked | Use native interactive elements or add tabindex="0" and key handlers |
| Missing form labels | Screen readers cannot announce what an input is for | Associate every input with a <label> using for/id or wrapping |
| Auto-playing media | Disorienting for screen reader users, harmful for those with cognitive disabilities | Never auto-play; if unavoidable, provide a visible pause/stop control |
| Missing skip link | Keyboard users must tab through the entire nav on every page | Add a skip-to-main-content link as the first focusable element |
| No focus indicator | Keyboard users lose track of their position on the page | Never remove outline without providing a visible custom alternative |
| Missing page language | Screen readers may mispronounce content | Set lang attribute on <html> (e.g., lang="en") |
| Inaccessible dynamic content | Screen readers do not announce changes that happen after page load | Use ARIA live regions to announce dynamic updates |
| Missing document title | Screen reader users cannot identify the page when switching tabs | Set a unique, descriptive <title> for every page |
| Touch targets too small | Motor-impaired users cannot reliably tap small controls | Minimum 24x24 CSS pixels (AA), prefer 44x44 for comfortable interaction |
| Motion and animation | Can cause vestigo, nausea, or seizures | Respect prefers-reduced-motion media query; never flash more than 3 times per second |
Accessible Forms
Forms are where accessibility failures cause the most real-world harm -- users cannot complete purchases, registrations, or critical workflows.
Label Association
Every form control needs a programmatically associated label:
<!-- Explicit association -->
<label for="email">Email address</label>
<input type="email" id="email" name="email" required>
<!-- Implicit association (wrapping) -->
<label>
Email address
<input type="email" name="email" required>
</label>
Error Handling
- Display errors inline next to the relevant field, not only at the top of the form
- Use
aria-describedbyto associate error messages with their input - Use
aria-invalid="true"on fields that fail validation - Provide clear, specific error text ("Enter an email in the format name@example.com" not "Invalid input")
Required Fields
- Mark required fields with
requiredattribute (for native validation) oraria-required="true"(for custom validation) - Do not rely solely on color or an asterisk to indicate required status -- add text like "(required)"
Color and Contrast
Color must never be the only means of conveying information. Pair it with text, icons, patterns, or other visual cues.
Contrast ratios (WCAG AA):
| Element | Minimum Ratio |
|---|---|
| Normal text (under 18pt / 14pt bold) | 4.5:1 |
| Large text (18pt+ / 14pt+ bold) | 3:1 |
| UI components and graphical objects | 3:1 |
Quick wins:
- Test with browser developer tools (Chrome DevTools shows contrast ratios on hover)
- Respect
prefers-color-schemefor dark mode support - Test designs with simulated color blindness (protanopia, deuteranopia, tritanopia)
Reference Files
| Reference | Contents |
|---|---|
| ARIA Patterns | When to use ARIA, roles and properties, widget patterns (tabs, modals, accordions, dropdowns), live regions, landmark roles |
| Keyboard and Focus | Tab order, arrow key navigation, focus trapping, skip links, focus restoration, visible focus indicators, touch targets |
| Testing Strategies | Automated tools, manual testing checklist, screen reader testing, contrast verification, accessibility tree, CI integration |
Integration with Other Skills
| Situation | Recommended Skill |
|---|---|
| Building REST or GraphQL APIs with accessible error responses | Install knowledge-virtuoso from krzysztofsurdy/code-virtuoso for API design guidance |
| Security headers that affect accessibility (CSP, iframe restrictions) | Install knowledge-virtuoso from krzysztofsurdy/code-virtuoso for security patterns |
| Testing accessible components with unit and integration tests | Install knowledge-virtuoso from krzysztofsurdy/code-virtuoso for testing strategies |
| Performance optimization that does not sacrifice accessibility | Install knowledge-virtuoso from krzysztofsurdy/code-virtuoso for performance guidance |