docs/skills/epic-ui-guidelines/SKILL.md
Guide on UI/UX guidelines, accessibility, and component usage for Epic Stack
npx skillsauth add epicweb-dev/gratitext epic-ui-guidelinesInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Use this skill when you need to:
Following Epic Web principles:
Software is built for people, by people - Accessibility isn't about checking boxes or meeting standards. It's about creating software that works for real people with diverse needs, abilities, and contexts. Every UI decision should prioritize the human experience over technical convenience.
Accessibility is not optional - it's how we ensure our software serves all users, not just some. When you make UI accessible, you're making it better for everyone: clearer labels help all users, keyboard navigation helps power users, and semantic HTML helps search engines.
Example - Human-centered approach:
// ✅ Good - Built for people
function NoteForm() {
return (
<Form method="POST">
<Field
labelProps={{
htmlFor: fields.title.id,
children: 'Note Title', // Clear, human-readable label
}}
inputProps={{
...getInputProps(fields.title),
placeholder: 'Enter a descriptive title', // Helpful guidance
autoFocus: true, // Saves time for users
}}
errors={fields.title.errors} // Clear error messages
/>
</Form>
)
}
// ❌ Avoid - Technical convenience over user experience
function NoteForm() {
return (
<Form method="POST">
<input name="title" /> {/* No label, no guidance, no accessibility */}
</Form>
)
}
✅ Good - Use semantic elements:
function UserCard({ user }: { user: User }) {
return (
<article>
<header>
<h2>{user.name}</h2>
</header>
<p>{user.bio}</p>
<footer>
<time dateTime={user.createdAt}>{formatDate(user.createdAt)}</time>
</footer>
</article>
)
}
❌ Avoid - Generic divs:
// ❌ Don't use divs for everything
<div>
<div>{user.name}</div>
<div>{user.bio}</div>
<div>{formatDate(user.createdAt)}</div>
</div>
✅ Good - Always use labels:
import { Field } from '#app/components/forms.tsx'
<Field
labelProps={{
htmlFor: fields.email.id,
children: 'Email',
}}
inputProps={{
...getInputProps(fields.email, { type: 'email' }),
autoFocus: true,
autoComplete: 'email',
}}
errors={fields.email.errors}
/>
The Field component automatically:
htmlFor and idaria-invalid when there are errorsaria-describedby pointing to error messages❌ Avoid - Unlabeled inputs:
// ❌ Don't forget labels
<input type="email" name="email" />
✅ Good - Use ARIA appropriately:
// Epic Stack's Field component handles this automatically
<Field
inputProps={{
...getInputProps(fields.email, { type: 'email' }),
// aria-invalid and aria-describedby are added automatically
}}
errors={fields.email.errors} // Error messages are linked via aria-describedby
/>
✅ Good - ARIA for custom components:
function LoadingButton({ isLoading, children }: { isLoading: boolean; children: React.ReactNode }) {
return (
<button aria-busy={isLoading} disabled={isLoading}>
{isLoading ? 'Loading...' : children}
</button>
)
}
Epic Stack uses Radix UI for accessible, unstyled components.
✅ Good - Use Radix primitives:
import * as Dialog from '@radix-ui/react-dialog'
import { Button } from '#app/components/ui/button.tsx'
function MyDialog() {
return (
<Dialog.Root>
<Dialog.Trigger asChild>
<Button>Open Dialog</Button>
</Dialog.Trigger>
<Dialog.Portal>
<Dialog.Overlay className="fixed inset-0 bg-black/50" />
<Dialog.Content className="fixed top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 bg-white p-6">
<Dialog.Title>Dialog Title</Dialog.Title>
<Dialog.Description>Dialog description</Dialog.Description>
<Dialog.Close asChild>
<Button>Close</Button>
</Dialog.Close>
</Dialog.Content>
</Dialog.Portal>
</Dialog.Root>
)
}
Radix components automatically handle:
✅ Good - Use Tailwind utility classes:
function Card({ children }: { children: React.ReactNode }) {
return (
<div className="rounded-lg border border-gray-200 bg-white p-6 shadow-sm">
{children}
</div>
)
}
✅ Good - Use Tailwind responsive utilities:
<div className="grid grid-cols-1 gap-4 md:grid-cols-2 lg:grid-cols-3">
{items.map(item => (
<Card key={item.id}>{item.name}</Card>
))}
</div>
✅ Good - Use Tailwind dark mode:
<div className="bg-white text-gray-900 dark:bg-gray-800 dark:text-gray-100">
{content}
</div>
✅ Good - Display errors accessibly:
import { Field, ErrorList } from '#app/components/forms.tsx'
<Field
labelProps={{ htmlFor: fields.email.id, children: 'Email' }}
inputProps={getInputProps(fields.email, { type: 'email' })}
errors={fields.email.errors} // Errors are displayed below input
/>
<ErrorList errors={form.errors} id={form.errorId} /> // Form-level errors
Errors are automatically:
aria-describedby✅ Good - Visible focus indicators:
// Tailwind's default focus:ring handles this
<button className="focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-offset-2">
Click me
</button>
✅ Good - Focus on form errors:
import { useEffect, useRef } from 'react'
function FormWithErrorFocus() {
const firstErrorRef = useRef<HTMLInputElement>(null)
useEffect(() => {
if (actionData?.errors && firstErrorRef.current) {
firstErrorRef.current.focus()
}
}, [actionData?.errors])
return <Field inputProps={{ ref: firstErrorRef, ... }} />
}
✅ Good - Keyboard accessible components:
// Radix components handle keyboard navigation automatically
<Dialog.Trigger asChild>
<Button>Open</Button>
</Dialog.Trigger>
// Custom components should support keyboard
<button
onKeyDown={(e) => {
if (e.key === 'Enter' || e.key === ' ') {
handleClick()
}
}}
>
Custom Button
</button>
✅ Good - Use accessible color combinations:
// Use Tailwind's semantic colors that meet WCAG AA
<div className="bg-white text-gray-900"> // High contrast
<div className="text-blue-600 hover:text-blue-700"> // Accessible links
❌ Avoid - Low contrast text:
// ❌ Don't use low contrast
<div className="bg-gray-100 text-gray-200"> // Very low contrast
✅ Good - Mobile-first approach:
<div className="
flex flex-col gap-4
md:flex-row md:gap-8
lg:gap-12
">
{/* Content */}
</div>
✅ Good - Responsive typography:
<h1 className="text-2xl md:text-3xl lg:text-4xl">
Responsive Heading
</h1>
✅ Good - Accessible loading indicators:
import { useNavigation } from 'react-router'
function SubmitButton() {
const navigation = useNavigation()
const isSubmitting = navigation.state === 'submitting'
return (
<button
type="submit"
disabled={isSubmitting}
aria-busy={isSubmitting}
>
{isSubmitting ? 'Saving...' : 'Save'}
</button>
)
}
✅ Good - Decorative icons:
import { Icon } from '#app/components/ui/icon.tsx'
<button aria-label="Delete note">
<Icon name="trash" />
<span className="sr-only">Delete note</span>
</button>
✅ Good - Semantic icons:
<button>
<Icon name="check" aria-hidden="true" />
Save
</button>
✅ Good - Add skip to main content:
// In your root layout
<a href="#main-content" className="sr-only focus:not-sr-only focus:absolute focus:top-0 focus:left-0 focus:z-50 focus:p-4 focus:bg-blue-600 focus:text-white">
Skip to main content
</a>
<main id="main-content">
{/* Main content */}
</main>
✅ Good - Forms work without JavaScript:
// Conform forms work without JavaScript
<Form method="POST" {...getFormProps(form)}>
<Field {...props} />
<StatusButton type="submit">Submit</StatusButton>
</Form>
Forms automatically:
✅ Good - Use semantic HTML first:
// ✅ Semantic HTML provides context automatically
<nav aria-label="Main navigation">
<ul>
<li><a href="/">Home</a></li>
<li><a href="/about">About</a></li>
</ul>
</nav>
✅ Good - Announce dynamic content:
import { useNavigation } from 'react-router'
function SearchResults({ results }: { results: Result[] }) {
const navigation = useNavigation()
const isSearching = navigation.state === 'loading'
return (
<div
role="status"
aria-live="polite"
aria-atomic="true"
className="sr-only"
>
{isSearching ? 'Searching...' : `${results.length} results found`}
</div>
)
}
✅ Good - Live regions for important updates:
function ToastContainer({ toasts }: { toasts: Toast[] }) {
return (
<div aria-live="assertive" aria-atomic="true" className="sr-only">
{toasts.map(toast => (
<div key={toast.id} role="alert">
{toast.message}
</div>
))}
</div>
)
}
ARIA live region options:
aria-live="polite" - For non-critical updates (search results, status
messages)aria-live="assertive" - For critical updates (errors, confirmations)aria-atomic="true" - Screen reader reads entire region on updatearia-atomic="false" - Screen reader reads only changed parts✅ Good - Tab order follows visual order:
// Elements appear in logical tab order
<nav>
<a href="/">Home</a>
<a href="/about">About</a>
<a href="/contact">Contact</a>
</nav>
✅ Good - Keyboard shortcuts:
import { useEffect } from 'react'
function SearchDialog({ onClose }: { onClose: () => void }) {
useEffect(() => {
function handleKeyDown(e: KeyboardEvent) {
if (e.key === 'Escape') {
onClose()
}
}
window.addEventListener('keydown', handleKeyDown)
return () => window.removeEventListener('keydown', handleKeyDown)
}, [onClose])
return <Dialog>{/* content */}</Dialog>
}
✅ Good - Focus trap in modals:
// Radix Dialog automatically handles focus trap
<Dialog.Root>
<Dialog.Content>
{/* Focus is trapped inside dialog */}
<Dialog.Close>Close</Dialog.Close>
</Dialog.Content>
</Dialog.Root>
✅ Good - Focus on route changes:
import { useEffect } from 'react'
import { useNavigation } from 'react-router'
function RouteComponent() {
const navigation = useNavigation()
const mainRef = useRef<HTMLElement>(null)
useEffect(() => {
if (navigation.state === 'idle' && mainRef.current) {
mainRef.current.focus()
}
}, [navigation.state])
return (
<main ref={mainRef} tabIndex={-1}>
{/* Content */}
</main>
)
}
✅ Good - Focus on errors:
import { useEffect, useRef } from 'react'
function FormWithErrorFocus({ actionData }: Route.ComponentProps) {
const firstErrorRef = useRef<HTMLInputElement>(null)
useEffect(() => {
if (actionData?.errors && firstErrorRef.current) {
// Focus first error field
firstErrorRef.current.focus()
// Announce error
firstErrorRef.current.setAttribute('aria-invalid', 'true')
}
}, [actionData?.errors])
return <Field inputProps={{ ref: firstErrorRef, ... }} />
}
✅ Good - Readable text sizes:
// Use Tailwind's text size scale
<p className="text-base md:text-lg">Readable body text</p>
<h1 className="text-2xl md:text-3xl lg:text-4xl">Clear headings</h1>
✅ Good - Sufficient line height:
// Tailwind defaults provide good line height
<p className="leading-relaxed">Comfortable reading</p>
❌ Avoid - Small or hard-to-read text:
// ❌ Don't use very small text
<p className="text-xs">Hard to read</p>
✅ Good - Sufficient touch targets:
// Buttons should be at least 44x44px (touch target size)
<button className="min-h-[44px] min-w-[44px] px-4 py-2">
Click me
</button>
✅ Good - Spacing between interactive elements:
<div className="flex gap-4">
<Button>Save</Button>
<Button>Cancel</Button>
</div>
✅ Good - Use semantic HTML for dates/times:
<time dateTime={note.createdAt.toISOString()}>
{formatDate(note.createdAt)}
</time>
✅ Good - Use semantic HTML for numbers:
// Screen readers can pronounce numbers correctly
<p>Total: <span aria-label={`${count} items`}>{count}</span></p>
✅ Good - Language attributes:
// In root.tsx
<html lang="en">
<body>
{/* Content */}
</body>
</html>
✅ Good - Maintain contrast in dark mode:
// Ensure sufficient contrast in both modes
<div className="bg-white text-gray-900 dark:bg-gray-900 dark:text-gray-100">
{content}
</div>
✅ Good - Respect user preference:
// Epic Stack automatically handles theme preference
// Use semantic colors that work in both modes
<button className="bg-blue-600 text-white hover:bg-blue-700 dark:bg-blue-500 dark:hover:bg-blue-600">
Button
</button>
✅ Good - Respect reduced motion:
// Tailwind automatically respects prefers-reduced-motion
<div className="transition-transform duration-200 hover:scale-105 motion-reduce:transition-none">
{/* Animations disabled for users who prefer reduced motion */}
</div>
✅ Good - Use CSS for animations:
// ✅ CSS animations can be disabled via prefers-reduced-motion
<div className="animate-fade-in">
{/* Content */}
</div>
// ❌ JavaScript animations may not respect user preferences
Field component which includes
labels - helps all users, not just screen reader users<article>, <header>, <nav>,
etc. - helps all users understand content structuresr-only instead of
display: none for screen reader only contentaria-live for dynamic content
announcementsprefers-reduced-motion media query -
helps users with vestibular disordersapp/components/forms.tsx - Accessible form componentsapp/components/ui/ - Accessible UI components using Radixapp/styles/tailwind.css - Tailwind configurationtesting
Guide on testing with Vitest and Playwright for Epic Stack
testing
Guide on security practices including CSP, rate limiting, and session security for Epic Stack
development
Guide on routing with React Router and react-router-auto-routes for Epic Stack
development
Guide on React patterns, performance optimization, and code quality for Epic Stack