Loading...
Loading...
Guide on UI/UX guidelines, accessibility, and component usage for Epic Stack
npx skill4agent add rubenpenap/epic-stack-agent-skills epic-ui-guidelines// ✅ 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>
)
}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>
)
}// ❌ Don't use divs for everything
<div>
<div>{user.name}</div>
<div>{user.bio}</div>
<div>{formatDate(user.createdAt)}</div>
</div>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}
/>FieldhtmlForidaria-invalidaria-describedby// ❌ Don't forget labels
<input type="email" name="email" />// 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
/>function LoadingButton({ isLoading, children }: { isLoading: boolean; children: React.ReactNode }) {
return (
<button aria-busy={isLoading} disabled={isLoading}>
{isLoading ? 'Loading...' : children}
</button>
)
}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>
)
}function Card({ children }: { children: React.ReactNode }) {
return (
<div className="rounded-lg border border-gray-200 bg-white p-6 shadow-sm">
{children}
</div>
)
}<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><div className="bg-white text-gray-900 dark:bg-gray-800 dark:text-gray-100">
{content}
</div>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 errorsaria-describedby// 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>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, ... }} />
}// 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>// 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// ❌ Don't use low contrast
<div className="bg-gray-100 text-gray-200"> // Very low contrast<div className="
flex flex-col gap-4
md:flex-row md:gap-8
lg:gap-12
">
{/* Content */}
</div><h1 className="text-2xl md:text-3xl lg:text-4xl">
Responsive Heading
</h1>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>
)
}import { Icon } from '#app/components/ui/icon.tsx'
<button aria-label="Delete note">
<Icon name="trash" />
<span className="sr-only">Delete note</span>
</button><button>
<Icon name="check" aria-hidden="true" />
Save
</button>// 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>// Conform forms work without JavaScript
<Form method="POST" {...getFormProps(form)}>
<Field {...props} />
<StatusButton type="submit">Submit</StatusButton>
</Form>// ✅ 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>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>
)
}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="polite"aria-live="assertive"aria-atomic="true"aria-atomic="false"// Elements appear in logical tab order
<nav>
<a href="/">Home</a>
<a href="/about">About</a>
<a href="/contact">Contact</a>
</nav>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>
}// Radix Dialog automatically handles focus trap
<Dialog.Root>
<Dialog.Content>
{/* Focus is trapped inside dialog */}
<Dialog.Close>Close</Dialog.Close>
</Dialog.Content>
</Dialog.Root>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>
)
}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, ... }} />
}// 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>// Tailwind defaults provide good line height
<p className="leading-relaxed">Comfortable reading</p>// ❌ Don't use very small text
<p className="text-xs">Hard to read</p>// Buttons should be at least 44x44px (touch target size)
<button className="min-h-[44px] min-w-[44px] px-4 py-2">
Click me
</button><div className="flex gap-4">
<Button>Save</Button>
<Button>Cancel</Button>
</div><time dateTime={note.createdAt.toISOString()}>
{formatDate(note.createdAt)}
</time>// Screen readers can pronounce numbers correctly
<p>Total: <span aria-label={`${count} items`}>{count}</span></p>// In root.tsx
<html lang="en">
<body>
{/* Content */}
</body>
</html>// Ensure sufficient contrast in both modes
<div className="bg-white text-gray-900 dark:bg-gray-900 dark:text-gray-100">
{content}
</div>// 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>// 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>// ✅ CSS animations can be disabled via prefers-reduced-motion
<div className="animate-fade-in">
{/* Content */}
</div>
// ❌ JavaScript animations may not respect user preferencesField<article><header><nav>sr-onlydisplay: nonearia-liveprefers-reduced-motionapp/components/forms.tsxapp/components/ui/app/styles/tailwind.css