Button

Displays a button or a component that looks like a button.

import { IconArrowUp } from "~/components/icons"
import { Button } from "~/components/ui/button"

export function ButtonDemo() {
  return (
    <div class="flex flex-wrap items-center gap-2 md:flex-row">
      <Button variant="outline">Button</Button>
      <Button aria-label="Submit" size="icon" variant="outline">
        <IconArrowUp />
      </Button>
    </div>
  )
}

Installation

Usage

import { Button } from "~/components/ui/button";
<Button variant="outline">Button</Button>

Cursor

Tailwind v4 switched from cursor: pointer to cursor: default for the button component.

If you want to keep the cursor: pointer behavior, add the following code to your CSS file:

globals.css
@layer base {
button:not(:disabled),
[role="button"]:not(:disabled) {
cursor: pointer;
}
}

Examples

Size

import { IconArrowUp } from "~/components/icons"
import { Button } from "~/components/ui/button"

export function ButtonSizes() {
  return (
    <div class="flex flex-col items-start gap-8 sm:flex-row">
      <div class="flex items-start gap-2">
        <Button size="sm" variant="outline">
          Small
        </Button>
        <Button aria-label="Submit" size="icon-sm" variant="outline">
          <IconArrowUp />
        </Button>
      </div>
      <div class="flex items-start gap-2">
        <Button variant="outline">Default</Button>
        <Button aria-label="Submit" size="icon" variant="outline">
          <IconArrowUp />
        </Button>
      </div>
      <div class="flex items-start gap-2">
        <Button size="lg" variant="outline">
          Large
        </Button>
        <Button aria-label="Submit" size="icon-lg" variant="outline">
          <IconArrowUp />
        </Button>
      </div>
    </div>
  )
}

Default

import { Button } from "~/components/ui/button"

export function ButtonDefault() {
  return <Button>Button</Button>
}

Outline

import { Button } from "~/components/ui/button"

export function ButtonOutline() {
  return <Button variant="outline">Outline</Button>
}

Secondary

import { Button } from "~/components/ui/button"

export function ButtonSecondary() {
  return <Button variant="secondary">Secondary</Button>
}

Ghost

import { Button } from "~/components/ui/button"

export function ButtonGhost() {
  return <Button variant="ghost">Ghost</Button>
}

Destructive

import { Button } from "~/components/ui/button"

export function ButtonDestructive() {
  return <Button variant="destructive">Destructive</Button>
}
import { Button } from "~/components/ui/button"

export function ButtonLink() {
  return <Button variant="link">Link</Button>
}

Icon

import { IconDownload } from "~/components/icons"
import { Button } from "~/components/ui/button"

export function ButtonIcon() {
  return (
    <Button size="icon" variant="outline">
      <IconDownload />
    </Button>
  )
}

With Icon

The spacing between the icon and the text is automatically adjusted based on the size of the button. You do not need any margin on the icon.

import { IconGitBranch } from "~/components/icons"
import { Button } from "~/components/ui/button"

export function ButtonWithIcon() {
  return (
    <Button size="sm" variant="outline">
      <IconGitBranch /> New Branch
    </Button>
  )
}

Rounded

import { IconArrowUp } from "~/components/icons"
import { Button } from "~/components/ui/button"

export function ButtonRounded() {
  return (
    <Button class="rounded-full" size="icon" variant="outline">
      <IconArrowUp />
    </Button>
  )
}

Spinner

import { Button } from "~/components/ui/button"
import { Spinner } from "~/components/ui/spinner"

export function ButtonLoading() {
  return (
    <Button disabled size="sm" variant="outline">
      <Spinner /> Submit
    </Button>
  )
}

Button Group

To create a button group, use the ButtonGroup component. See the Button Group documentation for more details.

import { createSignal } from "solid-js"

import {
  IconArchive,
  IconArrowLeft,
  IconCalendarPlus,
  IconClock,
  IconDots,
  IconFilterPlus,
  IconMailCheck,
  IconTag,
  IconTrash
} from "~/components/icons"
import { Button } from "~/components/ui/button"
import { ButtonGroup } from "~/components/ui/button-group"
import {
  DropdownMenu,
  DropdownMenuContent,
  DropdownMenuGroup,
  DropdownMenuItem,
  DropdownMenuPortal,
  DropdownMenuRadioGroup,
  DropdownMenuRadioItem,
  DropdownMenuSeparator,
  DropdownMenuSub,
  DropdownMenuSubContent,
  DropdownMenuSubTrigger,
  DropdownMenuTrigger
} from "~/components/ui/dropdown-menu"

export function ButttonGroupDemo() {
  const [label, setLabel] = createSignal("personal")
  return (
    <ButtonGroup>
      <ButtonGroup class="hidden sm:flex">
        <Button aria-label="Go Back" size="icon" variant="outline">
          <IconArrowLeft />
        </Button>
      </ButtonGroup>
      <ButtonGroup>
        <Button variant="outline">Archive</Button>
        <Button variant="outline">Report</Button>
      </ButtonGroup>
      <ButtonGroup>
        <Button variant="outline">Snooze</Button>
        <DropdownMenu placement="bottom-end">
          <DropdownMenuTrigger
            aria-label="More Options"
            as={Button<"button">}
            size="icon"
            variant="outline"
          >
            <IconDots />
          </DropdownMenuTrigger>
          <DropdownMenuContent class="w-52">
            <DropdownMenuGroup>
              <DropdownMenuItem>
                <IconMailCheck />
                Mark as Read
              </DropdownMenuItem>
              <DropdownMenuItem>
                <IconArchive />
                Archive
              </DropdownMenuItem>
            </DropdownMenuGroup>
            <DropdownMenuSeparator />
            <DropdownMenuGroup>
              <DropdownMenuItem>
                <IconClock />
                Snooze
              </DropdownMenuItem>
              <DropdownMenuItem>
                <IconCalendarPlus />
                Add to Calendar
              </DropdownMenuItem>
              <DropdownMenuItem>
                <IconFilterPlus />
                Add to List
              </DropdownMenuItem>
              <DropdownMenuSub overlap>
                <DropdownMenuSubTrigger>
                  <IconTag />
                  Label As...
                </DropdownMenuSubTrigger>
                <DropdownMenuPortal>
                  <DropdownMenuSubContent>
                    <DropdownMenuRadioGroup onChange={setLabel} value={label()}>
                      <DropdownMenuRadioItem value="personal">Personal</DropdownMenuRadioItem>
                      <DropdownMenuRadioItem value="work">Work</DropdownMenuRadioItem>
                      <DropdownMenuRadioItem value="other">Other</DropdownMenuRadioItem>
                    </DropdownMenuRadioGroup>
                  </DropdownMenuSubContent>
                </DropdownMenuPortal>
              </DropdownMenuSub>
            </DropdownMenuGroup>
            <DropdownMenuSeparator />
            <DropdownMenuGroup>
              <DropdownMenuItem variant="destructive">
                <IconTrash />
                Trash
              </DropdownMenuItem>
            </DropdownMenuGroup>
          </DropdownMenuContent>
        </DropdownMenu>
      </ButtonGroup>
    </ButtonGroup>
  )
}

You can use the as prop to make another component look like a button. Here's an example of a link that looks like a button.

import { A } from "@solidjs/router";
import { Button } from "~/components/ui/button";
export function LinkAsButton() {
return (
<Button as={A} href="/login">
Login
</Button>
);
}

API Reference

Button

The Button component is a wrapper around the button element that adds a variety of styles and functionality.

PropTypeDefault
variant"default" | "outline" | "ghost" | "destructive" | "secondary" | "link""default"
size"default" | "sm" | "lg" | "icon" | "icon-sm" | "icon-lg""default"
asValidComponent