install
source · Clone the upstream repo
git clone https://github.com/ComeOnOliver/skillshub
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/ComeOnOliver/skillshub "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/TerminalSkills/skills/class-variance-authority" ~/.claude/skills/comeonoliver-skillshub-class-variance-authority && rm -rf "$T"
manifest:
skills/TerminalSkills/skills/class-variance-authority/SKILL.mdsource content
Class Variance Authority (CVA)
Overview
CVA creates type-safe component variants with Tailwind. Define base styles, variants (size, color, state), compound variants, and defaults. Works with any framework. Powers the variant system in shadcn/ui.
Instructions
Step 1: Define Variants
// components/button.tsx — Button with CVA variants import { cva, type VariantProps } from 'class-variance-authority' import { cn } from '@/lib/utils' const buttonVariants = cva( // Base styles — always applied 'inline-flex items-center justify-center rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 disabled:pointer-events-none disabled:opacity-50', { variants: { variant: { default: 'bg-blue-600 text-white hover:bg-blue-700', secondary: 'bg-gray-100 text-gray-900 hover:bg-gray-200', destructive: 'bg-red-600 text-white hover:bg-red-700', outline: 'border border-gray-300 bg-transparent hover:bg-gray-50', ghost: 'hover:bg-gray-100', link: 'text-blue-600 underline-offset-4 hover:underline', }, size: { sm: 'h-8 px-3 text-xs', md: 'h-10 px-4 text-sm', lg: 'h-12 px-6 text-base', icon: 'h-10 w-10', }, }, compoundVariants: [ // Destructive + outline = red border { variant: 'outline', className: 'border-2' }, { variant: 'destructive', size: 'lg', className: 'text-lg font-bold' }, ], defaultVariants: { variant: 'default', size: 'md', }, } ) interface ButtonProps extends React.ButtonHTMLAttributes<HTMLButtonElement>, VariantProps<typeof buttonVariants> {} function Button({ className, variant, size, ...props }: ButtonProps) { return <button className={cn(buttonVariants({ variant, size }), className)} {...props} /> }
Step 2: Complex Components
const badgeVariants = cva( 'inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-semibold', { variants: { status: { active: 'bg-green-100 text-green-800', inactive: 'bg-gray-100 text-gray-600', warning: 'bg-yellow-100 text-yellow-800', error: 'bg-red-100 text-red-800', }, size: { sm: 'text-[10px] px-1.5 py-0', md: 'text-xs px-2.5 py-0.5', lg: 'text-sm px-3 py-1', }, }, defaultVariants: { status: 'active', size: 'md' }, } ) // Usage <Badge status="active">Online</Badge> <Badge status="error" size="sm">Failed</Badge>
Guidelines
- CVA is for defining variants. Use
(clsx + tailwind-merge) for conditional classes.cn()
for combinations that need special treatment (e.g., destructive + outline).compoundVariants
gives you TypeScript types for free.VariantProps<typeof variants>- Keep CVA definitions close to their component — one file per component.
- Works with any CSS framework, but shines with Tailwind's utility classes.