shadcn-best-practices

📁 ofershap/shadcn-best-practices 📅 9 days ago
3
总安装量
3
周安装量
#57202
全站排名
安装命令
npx skills add https://github.com/ofershap/shadcn-best-practices --skill shadcn-best-practices

Agent 安装分布

cline 3
gemini-cli 3
codebuddy 3
codex 3
continue 3
cursor 3

Skill 文档

When to use

Use this skill when working with shadcn/ui. Agents often treat it as a conventional npm package, suggest wrong imports, skip the form patterns, and ignore the copy-paste architecture. This skill anchors correct component sourcing, form patterns with React Hook Form + Zod, theming via CSS variables, and composition patterns.

Critical Rules

1. Components live in your project, not a package

Wrong:

import { Button } from "@shadcn/ui";
import { Button } from "shadcn-ui";

Correct:

import { Button } from "@/components/ui/button";

Why: shadcn/ui is a collection of copy-paste components. You copy them into your project and own the source. There is no @shadcn/ui package to import from.

2. Use cn() for conditional classes

Wrong:

<div className={`flex ${variant === "primary" ? "bg-blue-500" : "bg-gray-500"} ${className}`}>

Correct:

import { cn } from "@/lib/utils"
<div className={cn("flex", variant === "primary" && "bg-blue-500", className)}>

Why: cn() from @/lib/utils merges Tailwind classes correctly and resolves conflicts. Template literals produce invalid combinations and unpredictable specificity.

3. Forms: React Hook Form + Zod + shadcn Form

Wrong:

<form
  onSubmit={(e) => {
    e.preventDefault(); /* manual validation */
  }}
>
  <input onChange={(e) => setValue(e.target.value)} />
</form>

Correct:

import { zodResolver } from "@hookform/resolvers/zod"
import { useForm } from "react-hook-form"
import { z } from "zod"
import { Form, FormField, FormItem, FormControl, FormMessage } from "@/components/ui/form"

const schema = z.object({ username: z.string().min(2) })
const form = useForm({ resolver: zodResolver(schema), defaultValues: { username: "" } })

<Form {...form}>
  <form onSubmit={form.handleSubmit(onSubmit)}>
    <FormField control={form.control} name="username" render={({ field }) => (
      <FormItem>
        <FormControl><Input {...field} /></FormControl>
        <FormMessage />
      </FormItem>
    )} />
  </form>
</Form>

Why: shadcn Form is built for React Hook Form with Zod. Manual validation and uncontrolled inputs bypass validation, a11y, and error display.

4. Build custom components with Radix primitives

Wrong:

<div onClick={() => setOpen(!open)} role="button">
  ...
</div>

Correct:

import * as Dialog from "@radix-ui/react-dialog";
<Dialog.Root>
  <Dialog.Trigger>...</Dialog.Trigger>...
</Dialog.Root>;

Why: Radix provides accessibility, keyboard handling, and focus management. Rebuilding from divs duplicates bugs and a11y gaps.

5. Theming via CSS variables

Wrong:

<Button className="bg-blue-500 text-white hover:bg-blue-600">

Correct:

/* globals.css */
:root {
  --primary: 222.2 84% 4.9%;
  --primary-foreground: 210 40% 98%;
}
<Button>

Why: shadcn components read from CSS variables. Hardcoding colors breaks theming and dark mode.

6. Accessibility

Wrong:

<div onClick={handleClick}>Click me</div>
<button>Submit</button>

Correct:

<button onClick={handleClick} aria-label="Open menu">Open</button>
<button type="submit">Submit</button>

Why: Buttons need proper roles, labels, and keyboard support. Use semantic elements and aria attributes.

7. Use cva for component variants

Wrong:

className={cn("base", size === "sm" && "text-sm", size === "lg" && "text-lg", variant === "outline" && "border", ...)}

Correct:

import { cva } from "class-variance-authority"
const variants = cva("base", { variants: { size: { sm: "text-sm", lg: "text-lg" }, variant: { outline: "border" } } })
className={variants({ size, variant })}

Why: cva keeps variant logic declarative and type-safe. Conditional chains are hard to maintain and error-prone.

8. Dialog/Sheet/Popover: controlled state and navigation

Wrong:

<Dialog open={open} onOpenChange={() => {}}>

Correct:

<Dialog open={open} onOpenChange={setOpen}>

Why: Always pass controlled state and handle close. Close on route change or escape via onOpenChange so state stays in sync.

9. Data tables with TanStack Table

Wrong:

<table>{data.map(row => <tr>...)}</table>

Correct:

import { useReactTable, getCoreRowModel } from "@tanstack/react-table";
import { Table, TableHeader, TableBody, TableRow } from "@/components/ui/table";
const table = useReactTable({ data, columns, getCoreRowModel: getCoreRowModel() });

Why: shadcn Table is designed to work with @tanstack/react-table for sorting, filtering, and virtualization.

10. Toast: use sonner integration

Wrong:

import { toast } from "custom-toast-library";
toast("Done");

Correct:

import { toast } from "sonner";
toast.success("Done");

Why: shadcn uses sonner for Toaster. Use the configured sonner instance to match app styling and placement.

11. Never import from @shadcn/ui

Wrong:

import { Component } from "@shadcn/ui";

Correct:

import { Component } from "@/components/ui/component";

Why: Components live in your repo at @/components/ui/. Add new ones with the CLI.

12. Add components with the CLI

Wrong:

npm install @shadcn/button

Correct:

npx shadcn@latest add button

Why: The CLI copies component source and configures your project. Install adds nothing useful.

Patterns

  • Keep component source in @/components/ui/ and customize it directly
  • Use FormField, FormItem, FormControl, FormMessage for every form field
  • Define theme in globals.css with –* CSS variables
  • Use Radix primitives when extending beyond existing shadcn components
  • Prefer cn() for any conditional or composed class names

Anti-Patterns

  • Importing from @shadcn/ui or any shadcn package
  • Wrapping shadcn components instead of editing source
  • Template literals for conditional Tailwind classes
  • Uncontrolled forms or manual validation instead of React Hook Form + Zod
  • Hardcoding hex/rgb colors in components
  • Building dialogs or dropdowns from divs instead of Radix
  • Custom toast implementations instead of sonner
  • Adding components via npm install instead of shadcn CLI