187 lines
6.3 KiB
TypeScript
187 lines
6.3 KiB
TypeScript
import * as React from "react"
|
|
import { Slot } from "@radix-ui/react-slot"
|
|
import { cva, type VariantProps } from "class-variance-authority"
|
|
import { color, shadow, rounded } from "@/lib/type"
|
|
|
|
import { cn } from "@/lib/utils"
|
|
|
|
const buttonVariants = cva(
|
|
"inline-flex items-center justify-center px-4 md:px-6 whitespace-nowrap rounded-md text-sm font-medium transition-all duration-300 hover:ring-2 hover:ring-offset-2 ring-offset-background transition-all focus-visible:outline-none focus-visible:hidden focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50",
|
|
{
|
|
variants: {
|
|
color: {
|
|
default: "bg-default text-default-foreground hover:bg-default/90 hover:ring-default dark:disabled:bg-default-500",
|
|
primary: "bg-primary text-primary-foreground hover:bg-primary/90 hover:ring-primary",
|
|
secondary: "bg-secondary text-secondary-foreground hover:ring-secondary",
|
|
destructive: "bg-destructive text-destructive-foreground hover:bg-destructive/90 hover:ring-destructive",
|
|
warning: "bg-warning text-warning-foreground hover:bg-warning/90 hover:ring-warning",
|
|
info: "bg-info text-info-foreground hover:bg-info/90 hover:ring-info",
|
|
success: "bg-success text-success-foreground hover:bg-success/90 hover:ring-success",
|
|
},
|
|
variant: {
|
|
default: "",
|
|
outline: "border border-default text-default bg-transparent hover:bg-default hover:text-default-foreground hover:ring-0 hover:ring-transparent",
|
|
soft: "text-default bg-default/10 hover:bg-default hover:text-default-foreground",
|
|
ghost: "text-default bg-transparent hover:bg-default hover:text-default-foreground hover:ring-0 hover:ring-transparent hover:ring-offset-0",
|
|
shadow: "shadow-md"
|
|
},
|
|
shadow: {
|
|
sm: "shadow",
|
|
md: "shadow-md",
|
|
lg: "shadow-lg",
|
|
xl: "shadow-xl",
|
|
},
|
|
rounded: {
|
|
sm: "rounded",
|
|
md: "rounded-md",
|
|
lg: "rounded-lg",
|
|
full: "rounded-full",
|
|
},
|
|
fullWidth: {
|
|
true: "w-full",
|
|
},
|
|
size: {
|
|
default: "h-11 md:px-6 px-4 ",
|
|
sm: "h-7 text-xs md:px-4 px-3",
|
|
md: "h-9",
|
|
lg: "h-12 px-8 text-base md:px-10 px-7",
|
|
icon: "h-10 w-10 p-0 md:px-0 flex justify-center items-center",
|
|
|
|
}
|
|
},
|
|
compoundVariants: [
|
|
{
|
|
variant: "outline",
|
|
color: "primary",
|
|
className: "text-primary border-primary hover:text-primary-foreground hover:border-primary hover:bg-primary",
|
|
},
|
|
{
|
|
variant: "outline",
|
|
color: "secondary",
|
|
className: "border-secondary hover:bg-secondary hover:text-secondary-foreground ",
|
|
},
|
|
{
|
|
variant: "outline",
|
|
color: "success",
|
|
className:
|
|
"text-success border-success hover:text-success-foreground hover:border-success hover:bg-success",
|
|
},
|
|
{
|
|
variant: "outline",
|
|
color: "info",
|
|
className:
|
|
"text-info border-info hover:text-info-foreground hover:border-info hover:bg-info",
|
|
},
|
|
{
|
|
variant: "outline",
|
|
color: "warning",
|
|
className:
|
|
"text-warning border-warning hover:text-warning-foreground hover:border-warning hover:bg-warning",
|
|
},
|
|
{
|
|
variant: "outline",
|
|
color: "destructive",
|
|
className: "text-destructive border-destructive hover:text-destructive-foreground hover:border-destructive hover:bg-destructive",
|
|
},
|
|
{
|
|
variant: "outline",
|
|
color: "info",
|
|
className: "text-info hover:text-info-foreground hover:border-info hover:bg-info",
|
|
},
|
|
{
|
|
variant: "soft",
|
|
color: "primary",
|
|
className: "text-primary bg-primary/10 hover:bg-primary hover:text-primary-foreground"
|
|
},
|
|
{
|
|
variant: "soft",
|
|
color: "secondary",
|
|
className: "bg-secondary/90 hover:bg-secondary/80 hover:text-secondary-foreground"
|
|
},
|
|
{
|
|
variant: "soft",
|
|
color: "success",
|
|
className: "text-success bg-success/10 hover:bg-success hover:text-success-foreground"
|
|
},
|
|
{
|
|
variant: "soft",
|
|
color: "info",
|
|
className: "text-info bg-info/10 hover:bg-info hover:text-info-foreground"
|
|
},
|
|
{
|
|
variant: "soft",
|
|
color: "warning",
|
|
className: "text-warning bg-warning/10 hover:bg-warning hover:text-warning-foreground"
|
|
},
|
|
{
|
|
variant: "soft",
|
|
color: "destructive",
|
|
className: "text-destructive bg-destructive/10 hover:bg-destructive hover:text-destructive-foreground"
|
|
},
|
|
{
|
|
variant: "ghost",
|
|
color: "primary",
|
|
className: "text-primary hover:bg-primary hover:text-primary-foreground ",
|
|
},
|
|
{
|
|
variant: "ghost",
|
|
color: "secondary",
|
|
className: "hover:text-secondary-foreground hover:bg-secondary text-secondary-foreground",
|
|
},
|
|
{
|
|
variant: "ghost",
|
|
color: "success",
|
|
className: "text-success hover:text-success-foreground hover:bg-success",
|
|
},
|
|
{
|
|
variant: "ghost",
|
|
color: "info",
|
|
className: "text-info hover:text-info-foreground hover:bg-info",
|
|
},
|
|
{
|
|
variant: "ghost",
|
|
color: "warning",
|
|
className: "text-warning hover:text-warning-foreground hover:bg-warning",
|
|
},
|
|
|
|
{
|
|
variant: "ghost",
|
|
color: "destructive",
|
|
className: "text-destructive hover:text-destructive-foreground hover:bg-destructive",
|
|
}
|
|
],
|
|
defaultVariants: {
|
|
variant: "default",
|
|
color: "default",
|
|
size: "default",
|
|
},
|
|
}
|
|
);
|
|
|
|
export interface ButtonProps
|
|
extends React.ButtonHTMLAttributes<HTMLButtonElement>,
|
|
VariantProps<typeof buttonVariants> {
|
|
asChild?: boolean
|
|
color?: color
|
|
shadow?: shadow
|
|
rounded?: rounded
|
|
fullWidth?: boolean
|
|
size?: "default" | "sm" | "md" | "lg" | "icon"
|
|
}
|
|
|
|
const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
|
|
({ className, variant, size, color, fullWidth, shadow, rounded, asChild = false, ...props }, ref) => {
|
|
const Comp = asChild ? Slot : "button"
|
|
return (
|
|
<Comp
|
|
className={cn(buttonVariants({ variant, size, fullWidth, rounded, color, shadow, className }))}
|
|
ref={ref}
|
|
{...props}
|
|
/>
|
|
)
|
|
}
|
|
)
|
|
Button.displayName = "Button"
|
|
|
|
export { Button, buttonVariants }
|