즉, variants는 축 하나씩 관리, compoundVariants는 조합 스타일 관리라고 보면 된다.
import { Slot } from "@radix-ui/react-slot";
import { cva, type VariantProps } from "class-variance-authority";
import { cn } from "@/lib/utils";
import * as React from "react";
const buttonVariants = cva(
"inline-flex items-center justify-center gap-2 whitespace-nowrap font-medium transition-all disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:ring-[3px] focus-visible:ring-ring/50 focus-visible:border-ring",
{
variants: {
// 색상 의도
intent: {
primary: "bg-primary",
neutral: "bg-secondary",
destructive: "bg-destructive",
},
// 표현 방식
appearance: {
solid: "",
outline: "border bg-background",
ghost: "hover:bg-accent hover:text-accent-foreground",
link: "underline-offset-4 hover:underline bg-transparent px-0",
},
// 크기
size: {
sm: "h-8 px-3 text-sm rounded-md",
md: "h-9 px-4 text-sm rounded-md",
lg: "h-10 px-6 text-base rounded-lg",
icon: "size-9 rounded-md",
},
},
// 조합 전용 스타일
compoundVariants: [
{
intent: "primary",
appearance: "solid",
class: "px-4 py-[11px] sm:py-[15px] rounded-lg font-semibold",
},
],
defaultVariants: {
intent: "primary",
appearance: "solid",
size: "md",
},
}
);
type ButtonProps = React.ComponentProps<"button"> &
VariantProps<typeof buttonVariants> & { asChild?: boolean };
export function Button({
className,
intent,
appearance,
size,
asChild = false,
...props
}: ButtonProps) {
const Comp = asChild ? Slot : "button";
return (
<Comp
className={cn(buttonVariants({ intent, appearance, size, className }))}
{...props}
/>
);
}
// 기본 Primary Solid (CTA)
<Button intent="primary" appearance="solid">
정산금 조회하기
</Button>
// Neutral Solid
<Button intent="neutral" appearance="solid">
다음
</Button>
// Outline
<Button intent="neutral" appearance="outline">
인증하기
</Button>
// Ghost
<Button intent="neutral" appearance="ghost">
더보기
</Button>
// Link
<Button intent="primary" appearance="link">
자세히 보기
</Button>
// Icon Only
<Button size="icon" intent="primary" appearance="solid">
<PlusIcon />
</Button>
intent: 버튼의 색상 의도 (primary, neutral, destructive 등)appearance: 버튼의 표현 방식 (solid, outline, ghost, link)size: 버튼의 크기 (sm, md, lg, icon)compoundVariants: 여러 조건이 동시에 만족할 때 적용되는 조합 전용 스타일