Button
Displays a button or a component that looks like a button.
Preview
import { Button } from "@/components/ui/Button";
import { MixerHorizontalIcon } from "@radix-ui/react-icons";
export function ButtonDemo() {
return (
<div className="flex flex-col gap-4 items-start">
<Button variant="default">Default</Button>
<Button variant="fill">Fill</Button>
<Button variant="pill">Pill</Button>
<Button variant="link">Link</Button>
<Button variant="menu">Menu</Button>
</div>
)
}Installation
Make sure that namespace is set in your component.json file. Namespace docs: Learn more about namespaces
pnpm dlx shadcn@latest add @aura/buttonManual
Install the following dependencies:
pnpm install @radix-ui/react-slot class-variance-authorityCopy and paste the class names utility into your utils/class-names.ts file.
import { clsx, type ClassValue } from "clsx"
import { twMerge } from "tailwind-merge"
export function cn(...inputs: ClassValue[]) {
return twMerge(clsx(inputs))
}
Copy and paste the Button component into your components/ui/Button.tsx file.
/**
* @description Displays a button or a component that looks like a button.
*/
import * as React from "react";
import { Slot } from "@radix-ui/react-slot";
import { cva, type VariantProps } from "class-variance-authority";
import { cn } from "@/utils/class-names";
const buttonVariants = cva("button", {
variants: {
variant: {
default: "button-fill",
fill: "button-fill",
pill: "button-pill border border-gray-6 text-gray-11 bg-gray-2 hover:bg-gray-3",
link: "button-link",
menu: "button-menu",
},
size: {
default: "h-4",
xs: "h-2.5",
sm: "h-3",
md: "h-4",
lg: "h-5",
xl: "h-6",
icon: "w-3 h-3 p-0",
},
},
defaultVariants: {
variant: "default",
size: "default",
},
});
interface ButtonProps
extends React.ComponentProps<"button">,
VariantProps<typeof buttonVariants> {
asChild?: boolean;
isDisabled?: boolean;
isLoading?: boolean;
isLoadingText?: string | React.ReactNode;
mode?: VariantProps<typeof buttonVariants>["variant"];
label?: string | React.ReactNode;
}
const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
(props: ButtonProps, ref) => {
const {
className,
variant,
mode,
size,
asChild = false,
isDisabled,
isLoading,
isLoadingText,
children,
label,
...rest
} = props;
const Comp = asChild ? Slot : "button";
const disabled = isDisabled || isLoading || props.disabled;
const effectiveVariant = variant ?? mode;
return (
<Comp
data-slot="button"
className={cn(
buttonVariants({ variant: effectiveVariant, size, className }),
disabled && "opacity-50 cursor-not-allowed"
)}
ref={ref}
disabled={disabled}
{...rest}
>
{asChild ? (
children
) : (
<>
{isLoading && isLoadingText ? isLoadingText : <>{label}{children}</>}
</>
)}
</Comp>
);
}
);
Button.displayName = "Button";
export { Button, buttonVariants };
export type { ButtonProps };
export default Button;Usage
Fill
export const Fill = () => <Button variant="fill">Fill Button</Button>;Pill
export const Pill = () => <Button variant="pill">Pill Button</Button>;Link
export const Link = () => <Button variant="link">Link Button</Button>;Menu
export const Menu = () => <Button variant="menu">Menu Button</Button>;Sizes
export const Sizes = () => (
<div className="flex flex-col gap-4 items-start">
<Button size="xs">Extra Small</Button>
<Button size="sm">Small</Button>
<Button size="md">Medium</Button>
<Button size="lg">Large</Button>
<Button size="xl">Extra Large</Button>
<Button size="icon"><MixerHorizontalIcon className="icon" /></Button>
</div>
)Disabled
export const Disabled = () => <Button isDisabled>Disabled Button</Button>;Loading
export const Loading = () => <Button isLoading>Loading Button...</Button>;States
export const States = () => (
<div className="flex flex-col gap-4 items-start">
<Button>Normal</Button>
<Button isDisabled>Disabled</Button>
<Button isLoading>Loading</Button>
</div>
)ModeProp
export const ModeProp = () => (
<div className="flex flex-col gap-4 items-start">
<Button mode="default">Default (mode)</Button>
<Button mode="fill">Fill (mode)</Button>
<Button mode="pill">Pill (mode)</Button>
<Button mode="link">Link (mode)</Button>
<Button mode="menu">Menu (mode)</Button>
</div>
)