git clone https://github.com/NeverSight/learn-skills.dev
T=$(mktemp -d) && git clone --depth=1 https://github.com/NeverSight/learn-skills.dev "$T" && mkdir -p ~/.claude/skills && cp -r "$T/data/skills-md/agents-inc/skills/web-ui-radix-ui" ~/.claude/skills/neversight-learn-skills-dev-web-ui-radix-ui && rm -rf "$T"
data/skills-md/agents-inc/skills/web-ui-radix-ui/SKILL.mdRadix UI Primitives
Quick Guide: Radix UI provides unstyled, accessible primitives for building design systems. Use compound component patterns (Root, Trigger, Content),
for polymorphism, andasChildattributes for animations. Focus on behavior and accessibility - defer styling decisions to your styling solution. Current: v1.4.x (May 2025) - Full React 19 and RSC compatibility with new preview primitives.data-state
<critical_requirements>
CRITICAL: Before Using This Skill
All code must follow project conventions in CLAUDE.md (kebab-case, named exports, import ordering,
, named constants)import type
(You MUST use compound component anatomy - Root, Trigger, Portal, Content, Close - for overlay components)
(You MUST use
and spread all props when using forwardRef
with custom components - unless using React 19+ where asChild
is a regular prop)ref
(You MUST use Portal for overlays to escape CSS stacking contexts and parent overflow constraints)
(You MUST provide accessible labels via Title/Description components or ARIA attributes - Dialog logs console errors for missing Title)
</critical_requirements>
Auto-detection: Radix UI, radix-ui, @radix-ui, Dialog, Dropdown, DropdownMenu, Select, Popover, Tooltip, Accordion, Tabs, AlertDialog, asChild, Slot, Portal, data-state, OneTimePasswordField, PasswordToggleField, unstable_Form, Form.Field, Form.Message
When to use:
- Building accessible overlay components (dialogs, popovers, dropdowns, tooltips)
- Creating compound component APIs with multiple coordinated parts
- Implementing keyboard navigation and focus management
- Needing polymorphic components via
patternasChild
When NOT to use:
- Pre-styled components desired (use a component library built on Radix)
- Simple components without complex interactions (use plain HTML)
- Non-React projects (Radix primitives are React-specific)
Package Installation:
# Recommended: Unified tree-shakeable package (prevents version conflicts) npm i radix-ui # Alternative: Individual packages npm i @radix-ui/react-dialog @radix-ui/react-dropdown-menu
Detailed Resources:
- For core code examples (Dialog, asChild, Slot), see examples/core.md
- For overlay patterns (AlertDialog, controlled Dialog), see examples/overlays.md
- For menu patterns (DropdownMenu, submenus), see examples/menus.md
- For animation patterns (data-state, CSS keyframes), see examples/animation.md
- For form patterns (Select), see examples/forms.md
- For navigation patterns (Accordion, Tabs), see examples/navigation.md
- For preview components (OneTimePasswordField, PasswordToggleField), see examples/preview.md
- For decision frameworks and anti-patterns, see reference.md
<philosophy>
Philosophy
Radix UI Primitives provide behavioral and accessibility foundations without imposing visual design. Each primitive handles:
- Accessibility: ARIA attributes, roles, keyboard navigation, focus management
- Behavior: Open/close state, dismissal patterns, collision detection
- Composition: Compound components that work together as coordinated systems
Radix is styling-agnostic: Apply styles via
className prop using your styling solution. The primitives expose data-state attributes for state-based styling.
Compound Component Model: Each primitive consists of multiple parts (Root, Trigger, Content, etc.) that share context. This enables flexible composition while maintaining coordinated behavior.
React 19 & RSC Support (v1.4.3): Full compatibility with React 19 and React Server Components. Enhanced keyboard handling avoids browser hotkey interference.
</philosophy><patterns>
Core Patterns
Pattern 1: Compound Component Anatomy
Radix primitives use a compound component pattern where multiple parts work together through shared context.
Standard Structure for Overlay Components
import { Dialog } from "radix-ui"; // Root provides context and state management // Trigger opens the dialog // Portal renders content outside React tree // Overlay covers the page // Content contains the dialog body // Close dismisses the dialog // Title and Description provide accessibility <Dialog.Root> <Dialog.Trigger>Open</Dialog.Trigger> <Dialog.Portal> <Dialog.Overlay className={className} /> <Dialog.Content className={className}> <Dialog.Title>Dialog Title</Dialog.Title> <Dialog.Description>Accessible description</Dialog.Description> {/* Dialog content */} <Dialog.Close>Close</Dialog.Close> </Dialog.Content> </Dialog.Portal> </Dialog.Root>
Why this structure: Root manages state and context, Portal escapes CSS stacking contexts, Overlay provides visual backdrop, Title/Description ensure screen reader accessibility
Pattern 2: Controlled vs Uncontrolled State
Radix primitives support both controlled and uncontrolled state patterns.
Uncontrolled (Radix Manages State)
// Let Radix manage internal state - simpler for basic use cases <Dialog.Root defaultOpen={false}> <Dialog.Trigger>Open</Dialog.Trigger> <Dialog.Portal> <Dialog.Content> {/* Content */} </Dialog.Content> </Dialog.Portal> </Dialog.Root>
When to use: Simple dialogs without external state requirements
Controlled (You Manage State)
import { useState } from "react"; import { Dialog } from "radix-ui"; function ControlledDialog() { const [open, setOpen] = useState(false); const handleSave = async () => { await saveData(); setOpen(false); // Programmatically close after async operation }; return ( <Dialog.Root open={open} onOpenChange={setOpen}> <Dialog.Trigger>Open</Dialog.Trigger> <Dialog.Portal> <Dialog.Content> <Dialog.Title>Edit Profile</Dialog.Title> <button onClick={handleSave}>Save</button> </Dialog.Content> </Dialog.Portal> </Dialog.Root> ); }
When to use: Programmatic control needed (close after async, open from external trigger, sync with URL state)
Pattern 3: asChild for Polymorphism
The
asChild prop enables Radix to merge behavior onto your custom components or different element types.
Changing Element Type
import { Tooltip } from "radix-ui"; // Tooltip trigger defaults to button, but you may want a link <Tooltip.Root> <Tooltip.Trigger asChild> <a href="/docs">Documentation</a> </Tooltip.Trigger> <Tooltip.Portal> <Tooltip.Content>View the docs</Tooltip.Content> </Tooltip.Portal> </Tooltip.Root>
Why good: Radix passes all required props and event handlers to the anchor, maintaining accessibility
With Custom Components
import { forwardRef } from "react"; import { Dialog } from "radix-ui"; // Custom component MUST use forwardRef and spread props const CustomButton = forwardRef<HTMLButtonElement, React.ComponentProps<"button">>( ({ className, ...props }, ref) => { return <button ref={ref} className={className} {...props} />; } ); CustomButton.displayName = "CustomButton"; // Use with asChild <Dialog.Trigger asChild> <CustomButton className="custom-class">Open Dialog</CustomButton> </Dialog.Trigger>
Why this works: forwardRef allows Radix to attach refs for positioning/focus, spreading props passes event handlers and ARIA attributes
Pattern 4: Building Custom asChild Components with Slot
Use the
Slot utility to build your own components with asChild support.
import { forwardRef } from "react"; import { Slot } from "radix-ui"; export type ButtonProps = React.ComponentProps<"button"> & { asChild?: boolean; }; export const Button = forwardRef<HTMLButtonElement, ButtonProps>( ({ asChild = false, className, ...props }, ref) => { const Comp = asChild ? Slot : "button"; return <Comp ref={ref} className={className} {...props} />; } ); Button.displayName = "Button"; // Usage - renders as button <Button>Click me</Button> // Usage with asChild - renders as anchor <Button asChild> <a href="/page">Navigate</a> </Button>
Why good: Slot merges all props onto the child element, eliminating wrapper elements while preserving behavior
Pattern 5: Portal Usage for Overlays
Portal renders content outside the React component tree to escape CSS stacking contexts.
import { Popover } from "radix-ui"; <Popover.Root> <Popover.Trigger>Toggle Popover</Popover.Trigger> <Popover.Portal> {/* Rendered in document.body, escaping parent overflow:hidden */} <Popover.Content className={className}> <Popover.Arrow /> Popover content </Popover.Content> </Popover.Portal> </Popover.Root>
When to use: All overlay components (dialogs, popovers, tooltips, dropdown menus)
Custom Portal Container
import { useRef } from "react"; import { Dialog } from "radix-ui"; function DialogWithCustomContainer() { const containerRef = useRef<HTMLDivElement>(null); return ( <> <div ref={containerRef} /> <Dialog.Root> <Dialog.Trigger>Open</Dialog.Trigger> <Dialog.Portal container={containerRef.current}> <Dialog.Content>Content in custom container</Dialog.Content> </Dialog.Portal> </Dialog.Root> </> ); }
When to use: Micro-frontends, iframes, or specific DOM hierarchy requirements
Pattern 6: Animation with data-state Attributes
Radix primitives expose
data-state attributes for CSS-based animations. The unmount is suspended while exit animations complete. Use CSS @keyframes (not transition) -- Radix detects animation end events.
/* CSS keyframes — Radix suspends unmount until animation completes */ .dialog-overlay[data-state="open"] { animation: fadeIn 150ms ease-out; } .dialog-overlay[data-state="closed"] { animation: fadeOut 150ms ease-in; }
Critical: CSS
transition does NOT delay unmount -- only @keyframes animation works for exit animations.
JavaScript Animation Libraries
For complex orchestrated animations, use
forceMount on Portal, Overlay, and Content to prevent Radix from unmounting during exit animations. Wrap with your animation library's presence detection.
// Key pattern: controlled state + forceMount + conditional rendering <Dialog.Root open={open} onOpenChange={setOpen}> {open && ( <Dialog.Portal forceMount> <Dialog.Overlay asChild forceMount>{/* animated overlay */}</Dialog.Overlay> <Dialog.Content asChild forceMount>{/* animated content */}</Dialog.Content> </Dialog.Portal> )} </Dialog.Root>
See examples/animation.md for complete CSS keyframe and accordion height animation examples.
Pattern 7: Focus Management
Radix handles focus automatically for accessible interactions.
Default Behavior
// Focus automatically trapped in modal dialogs // Focus returns to trigger on close <Dialog.Root> <Dialog.Trigger>Open</Dialog.Trigger> <Dialog.Portal> <Dialog.Content> {/* Focus trapped here until closed */} <input autoFocus /> {/* Receives focus on open */} </Dialog.Content> </Dialog.Portal> </Dialog.Root>
Custom Focus Control
import { useRef } from "react"; import { AlertDialog } from "radix-ui"; function AlertDialogWithCustomFocus() { const cancelRef = useRef<HTMLButtonElement>(null); return ( <AlertDialog.Root> <AlertDialog.Trigger>Delete</AlertDialog.Trigger> <AlertDialog.Portal> <AlertDialog.Content onOpenAutoFocus={(e) => { e.preventDefault(); cancelRef.current?.focus(); // Focus cancel instead of first element }} > <AlertDialog.Title>Confirm Delete</AlertDialog.Title> <AlertDialog.Cancel ref={cancelRef}>Cancel</AlertDialog.Cancel> <AlertDialog.Action>Delete</AlertDialog.Action> </AlertDialog.Content> </AlertDialog.Portal> </AlertDialog.Root> ); }
Why custom focus: Destructive dialogs should focus the safe action (Cancel) by default
Pattern 8: Accessible Labels
Radix provides Title and Description components for screen reader accessibility.
import { Dialog } from "radix-ui"; <Dialog.Root> <Dialog.Trigger>Settings</Dialog.Trigger> <Dialog.Portal> <Dialog.Content aria-describedby={undefined} // Remove if no description > {/* Title is announced when dialog opens */} <Dialog.Title>Account Settings</Dialog.Title> {/* Description provides additional context */} <Dialog.Description> Manage your account preferences and security settings. </Dialog.Description> {/* Or visually hide but keep accessible */} <VisuallyHidden asChild> <Dialog.Description> This description is read by screen readers but not visible. </Dialog.Description> </VisuallyHidden> </Dialog.Content> </Dialog.Portal> </Dialog.Root>
Why mandatory: Screen readers announce Title when dialog opens, Description provides context for the interaction
</patterns><integration>
Integration Guide
Radix is behavior-only: Components are unstyled. Apply styles via
className prop using your styling solution.
Works with:
- Slot utility: Build custom
components with theasChild
component fromSlotradix-ui - Any CSS solution: Styles applied via className prop
- Animation libraries: Use
for JavaScript animation controlforceMount
Common Component Pairs:
| Primitive | Use Case |
|---|---|
| Dialog | Modal dialogs, forms, confirmations |
| AlertDialog | Destructive confirmations requiring explicit action |
| DropdownMenu | Navigation menus, action menus |
| Select | Form selects with custom styling |
| Popover | Non-modal floating content |
| Tooltip | Contextual information on hover/focus |
| Accordion | Expandable content sections |
| Tabs | Tabbed interfaces |
| Progress | Progress bars (supports for indeterminate) |
Preview Components (Unstable API):
| Primitive | Use Case | Import Prefix | Version |
|---|---|---|---|
| OneTimePasswordField | OTP input with keyboard nav, paste, autofill | | 0.1.8 |
| PasswordToggleField | Password visibility toggle with focus management | | 0.1.3 |
| Form | Form validation with constraint API | | 0.1.8 |
Note: Preview components use
unstable_ prefix. APIs may change before stable release.
</integration>
<red_flags>
RED FLAGS
High Priority Issues:
- Missing
on customforwardRef
components -- Radix cannot attach refs for positioning and focus managementasChild - Not spreading props on
components -- ARIA attributes and event handlers are lostasChild - Missing Portal for overlays -- content clipped by parent
or z-index issuesoverflow: hidden - Missing Title/Description on dialogs -- screen readers have no context (Dialog logs console errors)
- Using Dialog for destructive confirmations -- use AlertDialog (prevents accidental dismissal)
Gotchas & Edge Cases:
- CSS
does NOT delay unmount -- onlytransition
animation works for exit@keyframes
changes to "closed" before exit animation startsdata-state- AlertDialog requires Cancel or Action to close (no click-outside dismiss by design)
- React 19:
wrapper no longer needed --forwardRef
is a regular propref - Prefer unified
package over individualradix-ui
packages to prevent version conflicts@radix-ui/*
See reference.md for full anti-pattern examples with code and decision frameworks.
</red_flags>
<critical_reminders>
CRITICAL REMINDERS
All code must follow project conventions in CLAUDE.md
(You MUST use compound component anatomy - Root, Trigger, Portal, Content, Close - for overlay components)
(You MUST use
and spread all props when using forwardRef
with custom components - unless using React 19+ where asChild
is a regular prop)ref
(You MUST use Portal for overlays to escape CSS stacking contexts and parent overflow constraints)
(You MUST provide accessible labels via Title/Description components or ARIA attributes - Dialog logs console errors for missing Title)
Failure to follow these rules will break accessibility, focus management, and proper DOM rendering.
</critical_reminders>