- Accordion
- Alert
- Alert Dialog
- Aspect Ratio
- Avatar
- Badge
- Breadcrumb
- Button
- Button Group
- Calendar
- Card
- Carousel
- Chart
- Checkbox
- Collapsible
- Combobox
- Command
- Context Menu
- Data Table
- Date Picker
- Dialog
- Drawer
- Dropdown Menu
- Empty
- Fancy Button
- Field
- Filter
- Form
- Hover Card
- Input
- Input Group
- Input OTP
- Item
- Kbd
- Label
- Menubar
- Native Select
- Navigation Menu
- Number Field
- Pagination
- Pin Input
- Popover
- Progress
- Radio Group
- Radio Stack
- Range Calendar
- Range Slider
- Rating
- Resizable
- Scroll Area
- Select
- Separator
- Sheet
- Sidebar
- Skeleton
- Slider
- Sonner
- Spinner
- Stepper
- Swatch
- Swatch Group
- Switch
- Table
- Tabs
- Tags Input
- Textarea
- Toast
- Toggle
- Toggle Group
- Tooltip
- Typography
<script setup lang="ts">
import { ArrowUpIcon } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
</script>
<template>
<div class="flex flex-wrap items-center gap-2 md:flex-row">
<Button variant="outline">
Button
</Button>
<Button variant="outline" size="icon" aria-label="Submit">
<ArrowUpIcon />
</Button>
</div>
</template>Installation
pnpm dlx @frontic/ui add button
Usage
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="outline">
Button
</Button>
</template>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:
@layer base {
button:not(:disabled),
[role="button"]:not(:disabled) {
cursor: pointer;
}
}Frontic UI Extensions
Frontic UI extends this component with a variant/color separation architecture, e-commerce colors (buy, checkout), additional variants (subtle, form), and extended sizes (xs, xl).
Install the extended version:
pnpm dlx @frontic/ui add button
The Frontic Button uses a two-dimensional styling system:
variant: Controls the button style (default, subtle, outline, ghost, link, form, icon)color: Controls the color scheme (default, primary, secondary, neutral, inverted, buy, checkout, destructive, positive)
Colors
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<div class="flex flex-col gap-4">
<div class="flex flex-wrap items-center gap-2">
<Button color="default">Default</Button>
<Button color="primary">Primary</Button>
<Button color="secondary">Secondary</Button>
<Button color="neutral">Neutral</Button>
<Button color="inverted">Inverted</Button>
</div>
<div class="flex flex-wrap items-center gap-2">
<Button color="destructive">Destructive</Button>
<Button color="positive">Positive</Button>
<Button color="buy">Buy</Button>
<Button color="checkout">Checkout</Button>
</div>
</div>
</template><template>
<Button color="primary">Primary</Button>
<Button color="buy">Buy</Button>
<Button color="checkout">Checkout</Button>
</template>Buy Color (E-commerce)
The buy color is optimized for add-to-cart actions with enhanced visual feedback including shadow and scale effects.
<script setup lang="ts">
import { ShoppingCart } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
</script>
<template>
<div class="flex flex-wrap items-center gap-4">
<Button color="buy">
<ShoppingCart />
Add to Cart
</Button>
<Button color="buy" variant="outline">
<ShoppingCart />
Add to Cart
</Button>
<Button color="buy" variant="ghost">
<ShoppingCart />
Add to Cart
</Button>
</div>
</template><template>
<Button color="buy">
<ShoppingCart />
Add to Cart
</Button>
</template>Checkout Color (E-commerce)
The checkout color is designed for checkout and payment actions.
<script setup lang="ts">
import { CreditCard } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
</script>
<template>
<div class="flex flex-wrap items-center gap-4">
<Button color="checkout">
<CreditCard />
Checkout Now
</Button>
<Button color="checkout" variant="outline">
<CreditCard />
Checkout Now
</Button>
<Button color="checkout" variant="ghost">
<CreditCard />
Checkout Now
</Button>
</div>
</template><template>
<Button color="checkout">
<CreditCard />
Checkout Now
</Button>
</template>Subtle Variant
The subtle variant provides a softer appearance with reduced opacity backgrounds.
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<div class="flex flex-wrap items-center gap-4">
<Button variant="subtle">Default</Button>
<Button variant="subtle" color="primary">Primary</Button>
<Button variant="subtle" color="secondary">Secondary</Button>
<Button variant="subtle" color="destructive">Destructive</Button>
<Button variant="subtle" color="positive">Positive</Button>
</div>
</template><template>
<Button variant="subtle" color="primary">Primary</Button>
<Button variant="subtle" color="destructive">Destructive</Button>
</template>Extended Sizes
Frontic adds xs (extra small) and xl (extra large, rounded) sizes.
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<div class="flex flex-wrap items-center gap-4">
<Button size="xs">Extra Small</Button>
<Button size="sm">Small</Button>
<Button size="default">Default</Button>
<Button size="lg">Large</Button>
<Button size="xl">Extra Large</Button>
</div>
</template><template>
<Button size="xs">Extra Small</Button>
<Button size="xl">Extra Large</Button>
</template>Examples
Size
<script setup lang="ts">
import { ArrowUpRightIcon } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
</script>
<template>
<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 size="icon-sm" aria-label="Submit" variant="outline">
<ArrowUpRightIcon />
</Button>
</div>
<div class="flex items-start gap-2">
<Button variant="outline">
Default
</Button>
<Button size="icon" aria-label="Submit" variant="outline">
<ArrowUpRightIcon />
</Button>
</div>
<div class="flex items-start gap-2">
<Button variant="outline" size="lg">
Large
</Button>
<Button size="icon-lg" aria-label="Submit" variant="outline">
<ArrowUpRightIcon />
</Button>
</div>
</div>
</template>Default
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button>
Default
</Button>
</template>Outline
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="outline">
Outline
</Button>
</template>Secondary
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="secondary">
Secondary
</Button>
</template>Ghost
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="ghost">
Ghost
</Button>
</template>Destructive
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="destructive">
Destructive
</Button>
</template>Link
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="link">
Link
</Button>
</template>Icon
<script setup lang="ts">
import { CircleFadingArrowUpIcon } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="outline" size="icon">
<CircleFadingArrowUpIcon />
</Button>
</template><template>
<Button variant="outline" size="icon" aria-label="Submit">
<CircleFadingArrowUpIcon />
</Button>
</template>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.
<script setup lang="ts">
import { GitBranchIcon } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
</script>
<template>
<Button variant="outline" size="sm">
<GitBranchIcon />
New Branch
</Button>
</template>Rounded
Use the rounded-full class to make the button rounded.
<script setup lang="ts">
import { ArrowUpIcon } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
</script>
<template>
<div class="flex flex-col gap-8">
<Button variant="outline" size="icon" class="rounded-full">
<ArrowUpIcon />
</Button>
</div>
</template>Spinner
<script setup lang="ts">
import { Button } from '@/components/ui/button'
import { Spinner } from '@/components/ui/spinner'
</script>
<template>
<Button size="sm" variant="outline" disabled>
<Spinner class="animate-spin" />
Submit
</Button>
</template>Button Group
<script setup lang="ts">
import { ArchiveIcon, ArrowLeftIcon, CalendarPlusIcon, ClockIcon, ListFilterPlusIcon, MailCheckIcon, MoreHorizontalIcon, TagIcon, Trash2Icon } from 'lucide-vue-next'
import { Button } from '@/components/ui/button'
import { ButtonGroup } from '@/components/ui/button-group'
import { DropdownMenu, DropdownMenuContent, DropdownMenuGroup, DropdownMenuItem, DropdownMenuRadioGroup, DropdownMenuRadioItem, DropdownMenuSeparator, DropdownMenuSub, DropdownMenuSubContent, DropdownMenuSubTrigger, DropdownMenuTrigger } from '@/components/ui/dropdown-menu'
const label = ref('personal')
</script>
<template>
<ButtonGroup>
<ButtonGroup class="hidden sm:flex">
<Button variant="outline" size="icon" aria-label="Go Back">
<ArrowLeftIcon />
</Button>
</ButtonGroup>
<ButtonGroup>
<Button variant="outline">
Archive
</Button>
<Button variant="outline">
Report
</Button>
</ButtonGroup>
<ButtonGroup>
<Button variant="outline">
Snooze
</Button>
<DropdownMenu>
<DropdownMenuTrigger as-child>
<Button variant="outline" size="icon" aria-label="More Options">
<MoreHorizontalIcon />
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end" class="w-52">
<DropdownMenuGroup>
<DropdownMenuItem>
<MailCheckIcon />
Mark as Read
</DropdownMenuItem>
<DropdownMenuItem>
<ArchiveIcon />
Archive
</DropdownMenuItem>
</DropdownMenuGroup>
<DropdownMenuSeparator />
<DropdownMenuGroup>
<DropdownMenuItem>
<ClockIcon />
Snooze
</DropdownMenuItem>
<DropdownMenuItem>
<CalendarPlusIcon />
Add to Calendar
</DropdownMenuItem>
<DropdownMenuItem>
<ListFilterPlusIcon />
Add to List
</DropdownMenuItem>
<DropdownMenuSub>
<DropdownMenuSubTrigger>
<TagIcon class="mr-2 size-4" />
Label As...
</DropdownMenuSubTrigger>
<DropdownMenuSubContent>
<DropdownMenuRadioGroup v-model="label">
<DropdownMenuRadioItem value="personal">
Personal
</DropdownMenuRadioItem>
<DropdownMenuRadioItem value="work">
Work
</DropdownMenuRadioItem>
<DropdownMenuRadioItem value="other">
Other
</DropdownMenuRadioItem>
</DropdownMenuRadioGroup>
</DropdownMenuSubContent>
</DropdownMenuSub>
</DropdownMenuGroup>
<DropdownMenuSeparator />
<DropdownMenuGroup>
<DropdownMenuItem variant="destructive">
<Trash2Icon />
Trash
</DropdownMenuItem>
</DropdownMenuGroup>
</DropdownMenuContent>
</DropdownMenu>
</ButtonGroup>
</ButtonGroup>
</template>To create a button group, use the ButtonGroup component. See the Button Group documentation for more details.
Link (asChild)
You can use the as-child prop to make another component look like a button. Here's an example of a link that looks like a button.
<script setup lang="ts">
import { Button } from '@/components/ui/button'
</script>
<template>
<Button as-child>
<a href="/login">Login</a>
</Button>
</template>API Reference
Button
The Button component is a wrapper around the button element that adds a variety of styles and functionality.
| Prop | Type | Default |
|---|---|---|
variant | "default" | "subtle" | "outline" | "ghost" | "link" | "form" | "icon" | "secondary" | "destructive" | "brand" | "success" | "warning" | "info" | "default" |
color | "default" | "inverted" | "primary" | "secondary" | "neutral" | "buy" | "checkout" | "destructive" | "positive" | "default" |
size | "default" | "xs" | "sm" | "lg" | "xl" | "icon" | "icon-sm" | "icon-lg" | "default" |
asChild | boolean | false |
Note: For backwards compatibility, variant accepts color values like "secondary", "destructive", "brand", etc. The recommended approach is to use the color prop for semantic colors.