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 }