Base Components
Button
Renders a button or an element styled to resemble a button.
Installation
Install the following dependencies:
terminal
npm install @radix-ui/react-slot
Create a button.tsx
file and paste the following code into it.
/components/ui/button.tsx
// AlignUI Button v0.0.0
import * as React from 'react';
import { Slot } from '@radix-ui/react-slot';
import type { PolymorphicComponentProps } from '@/utils/polymorphic';
import { recursiveCloneChildren } from '@/utils/recursive-clone-children';
import { tv, type VariantProps } from '@/utils/tv';
const BUTTON_ROOT_NAME = 'ButtonRoot';
const BUTTON_ICON_NAME = 'ButtonIcon';
export const buttonVariants = tv({
slots: {
root: [
// base
'group relative inline-flex items-center justify-center whitespace-nowrap outline-none',
'transition duration-200 ease-out',
// focus
'focus:outline-none',
// disabled
'disabled:pointer-events-none disabled:bg-bg-weak-50 disabled:text-text-disabled-300 disabled:ring-transparent',
],
icon: [
// base
'flex size-5 shrink-0 items-center justify-center',
],
},
variants: {
variant: {
primary: {},
neutral: {},
error: {},
},
mode: {
filled: {},
stroke: {
root: 'ring-1 ring-inset',
},
lighter: {
root: 'ring-1 ring-inset',
},
ghost: {
root: 'ring-1 ring-inset',
},
},
size: {
medium: {
root: 'h-10 gap-3 rounded-10 px-3.5 text-label-sm',
icon: '-mx-1',
},
small: {
root: 'h-9 gap-3 rounded-lg px-3 text-label-sm',
icon: '-mx-1',
},
xsmall: {
root: 'h-8 gap-2.5 rounded-lg px-2.5 text-label-sm',
icon: '-mx-1',
},
xxsmall: {
root: 'h-7 gap-2.5 rounded-lg px-2 text-label-sm',
icon: '-mx-1',
},
},
},
compoundVariants: [
//#region variant=primary
{
variant: 'primary',
mode: 'filled',
class: {
root: [
// base
'bg-primary-base text-static-white',
// hover
'hover:bg-primary-darker',
// focus
'focus-visible:shadow-button-primary-focus',
],
},
},
{
variant: 'primary',
mode: 'stroke',
class: {
root: [
// base
'bg-bg-white-0 text-primary-base ring-primary-base',
// hover
'hover:bg-primary-alpha-10 hover:ring-transparent',
// focus
'focus-visible:shadow-button-primary-focus',
],
},
},
{
variant: 'primary',
mode: 'lighter',
class: {
root: [
// base
'bg-primary-alpha-10 text-primary-base ring-transparent',
// hover
'hover:bg-bg-white-0 hover:ring-primary-base',
// focus
'focus-visible:bg-bg-white-0 focus-visible:shadow-button-primary-focus focus-visible:ring-primary-base',
],
},
},
{
variant: 'primary',
mode: 'ghost',
class: {
root: [
// base
'bg-transparent text-primary-base ring-transparent',
// hover
'hover:bg-primary-alpha-10',
// focus
'focus-visible:bg-bg-white-0 focus-visible:shadow-button-primary-focus focus-visible:ring-primary-base',
],
},
},
//#endregion
//#region variant=neutral
{
variant: 'neutral',
mode: 'filled',
class: {
root: [
// base
'bg-bg-strong-950 text-text-white-0',
// hover
'hover:bg-bg-surface-800',
// focus
'focus-visible:shadow-button-important-focus',
],
},
},
{
variant: 'neutral',
mode: 'stroke',
class: {
root: [
// base
'bg-bg-white-0 text-text-sub-600 shadow-regular-xs ring-stroke-soft-200',
// hover
'hover:bg-bg-weak-50 hover:text-text-strong-950 hover:shadow-none hover:ring-transparent',
// focus
'focus-visible:text-text-strong-950 focus-visible:shadow-button-important-focus focus-visible:ring-stroke-strong-950',
],
},
},
{
variant: 'neutral',
mode: 'lighter',
class: {
root: [
// base
'bg-bg-weak-50 text-text-sub-600 ring-transparent',
// hover
'hover:bg-bg-white-0 hover:text-text-strong-950 hover:shadow-regular-xs hover:ring-stroke-soft-200',
// focus
'focus-visible:bg-bg-white-0 focus-visible:text-text-strong-950 focus-visible:shadow-button-important-focus focus-visible:ring-stroke-strong-950',
],
},
},
{
variant: 'neutral',
mode: 'ghost',
class: {
root: [
// base
'bg-transparent text-text-sub-600 ring-transparent',
// hover
'hover:bg-bg-weak-50 hover:text-text-strong-950',
// focus
'focus-visible:bg-bg-white-0 focus-visible:text-text-strong-950 focus-visible:shadow-button-important-focus focus-visible:ring-stroke-strong-950',
],
},
},
//#endregion
//#region variant=error
{
variant: 'error',
mode: 'filled',
class: {
root: [
// base
'bg-error-base text-static-white',
// hover
'hover:bg-red-700',
// focus
'focus-visible:shadow-button-error-focus',
],
},
},
{
variant: 'error',
mode: 'stroke',
class: {
root: [
// base
'bg-bg-white-0 text-error-base ring-error-base',
// hover
'hover:bg-red-alpha-10 hover:ring-transparent',
// focus
'focus-visible:shadow-button-error-focus',
],
},
},
{
variant: 'error',
mode: 'lighter',
class: {
root: [
// base
'bg-red-alpha-10 text-error-base ring-transparent',
// hover
'hover:bg-bg-white-0 hover:ring-error-base',
// focus
'focus-visible:bg-bg-white-0 focus-visible:shadow-button-error-focus focus-visible:ring-error-base',
],
},
},
{
variant: 'error',
mode: 'ghost',
class: {
root: [
// base
'bg-transparent text-error-base ring-transparent',
// hover
'hover:bg-red-alpha-10',
// focus
'focus-visible:bg-bg-white-0 focus-visible:shadow-button-error-focus focus-visible:ring-error-base',
],
},
},
//#endregion
],
defaultVariants: {
variant: 'primary',
mode: 'filled',
size: 'medium',
},
});
type ButtonSharedProps = VariantProps<typeof buttonVariants>;
type ButtonRootProps = VariantProps<typeof buttonVariants> &
React.ButtonHTMLAttributes<HTMLButtonElement> & {
asChild?: boolean;
};
const ButtonRoot = React.forwardRef<HTMLButtonElement, ButtonRootProps>(
(
{ children, variant, mode, size, asChild, className, ...rest },
forwardedRef,
) => {
const uniqueId = React.useId();
const Component = asChild ? Slot : 'button';
const { root } = buttonVariants({ variant, mode, size });
const sharedProps: ButtonSharedProps = {
variant,
mode,
size,
};
const extendedChildren = recursiveCloneChildren(
children as React.ReactElement[],
sharedProps,
[BUTTON_ICON_NAME],
uniqueId,
asChild,
);
return (
<Component
ref={forwardedRef}
className={root({ class: className })}
{...rest}
>
{extendedChildren}
</Component>
);
},
);
ButtonRoot.displayName = BUTTON_ROOT_NAME;
function ButtonIcon<T extends React.ElementType>({
variant,
mode,
size,
as,
className,
...rest
}: PolymorphicComponentProps<T, ButtonSharedProps>) {
const Component = as || 'div';
const { icon } = buttonVariants({ mode, variant, size });
return <Component className={icon({ class: className })} {...rest} />;
}
ButtonIcon.displayName = BUTTON_ICON_NAME;
export { ButtonRoot as Root, ButtonIcon as Icon };
Check the import paths to ensure they match your project setup.
Examples
Primary (Default)
Neutral
Error
Size
Disabled
With Icon
Full Width
asChild
Composition
You can simplify component usage by creating a custom API that abstracts the core functionalities.
API Reference
Button.Root
This component is based on the <button>
element and supports all of its props. And adds:
Prop | Type | Default |
---|---|---|
variant | "primary" |"neutral" |"error" | "primary" |
mode | "filled" |"stroke" |"lighter" |"ghost" | "filled" |
size | "medium" |"small" |"xsmall" |"xxsmall" | "medium" |
asChild | boolean |
Button.Icon
The Button.Icon
component is polymorphic, allowing you to change the underlying HTML element using the as
prop.
Prop | Type | Default |
---|---|---|
as | React.ElementType | div |
© 2024 AlignUI Design System. All rights reserved.
ON THIS PAGE