/ src / components / ui / button.tsx
button.tsx
 1  import * as React from "react"
 2  import { cva, type VariantProps } from "class-variance-authority"
 3  import { Slot } from "radix-ui"
 4  
 5  import { cn } from "@/lib/utils"
 6  
 7  const buttonVariants = cva(
 8    "inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm 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:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive",
 9    {
10      variants: {
11        variant: {
12          default: "bg-primary text-primary-foreground hover:bg-primary/90",
13          destructive:
14            "bg-destructive text-white hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
15          outline:
16            "border bg-background shadow-xs hover:bg-accent hover:text-accent-foreground dark:bg-input/30 dark:border-input dark:hover:bg-input/50",
17          secondary:
18            "bg-secondary text-secondary-foreground hover:bg-secondary/80",
19          ghost:
20            "hover:bg-accent hover:text-accent-foreground dark:hover:bg-accent/50",
21          link: "text-primary underline-offset-4 hover:underline",
22          accent:
23            "bg-accent-bright text-white rounded-[12px] font-600 hover:brightness-110 border-none",
24          surface:
25            "bg-transparent border border-white/[0.08] text-text-2 rounded-[12px] hover:bg-white/[0.04]",
26          "surface-icon":
27            "bg-white/[0.04] rounded-[10px] hover:bg-white/[0.06] border-none",
28        },
29        size: {
30          default: "h-9 px-4 py-2 has-[>svg]:px-3",
31          xs: "h-6 gap-1 rounded-md px-2 text-xs has-[>svg]:px-1.5 [&_svg:not([class*='size-'])]:size-3",
32          sm: "h-8 rounded-md gap-1.5 px-3 has-[>svg]:px-2.5",
33          lg: "h-10 rounded-md px-6 has-[>svg]:px-4",
34          icon: "size-9",
35          "icon-xs": "size-6 rounded-md [&_svg:not([class*='size-'])]:size-3",
36          "icon-sm": "size-8",
37          "icon-lg": "size-10",
38        },
39      },
40      defaultVariants: {
41        variant: "default",
42        size: "default",
43      },
44    }
45  )
46  
47  function Button({
48    className,
49    variant = "default",
50    size = "default",
51    asChild = false,
52    ...props
53  }: React.ComponentProps<"button"> &
54    VariantProps<typeof buttonVariants> & {
55      asChild?: boolean
56    }) {
57    const Comp = asChild ? Slot.Root : "button"
58  
59    return (
60      <Comp
61        data-slot="button"
62        data-variant={variant}
63        data-size={size}
64        className={cn(buttonVariants({ variant, size, className }))}
65        {...props}
66      />
67    )
68  }
69  
70  export { Button, buttonVariants }