2023-02-16 22:39:57 +00:00
|
|
|
import type { VariantProps } from "class-variance-authority";
|
|
|
|
import { cva } from "class-variance-authority";
|
|
|
|
import type { LinkProps } from "next/link";
|
|
|
|
import Link from "next/link";
|
2022-07-23 00:39:50 +00:00
|
|
|
import React, { forwardRef } from "react";
|
|
|
|
|
|
|
|
import classNames from "@calcom/lib/classNames";
|
2023-02-16 22:39:57 +00:00
|
|
|
import type { SVGComponent } from "@calcom/types/SVGComponent";
|
2023-01-26 22:51:03 +00:00
|
|
|
|
2023-04-12 15:26:31 +00:00
|
|
|
import { Plus } from "../icon";
|
2023-01-26 22:51:03 +00:00
|
|
|
import { Tooltip } from "../tooltip";
|
2022-07-23 00:39:50 +00:00
|
|
|
|
2022-11-23 02:55:25 +00:00
|
|
|
type InferredVariantProps = VariantProps<typeof buttonClasses>;
|
|
|
|
|
2023-01-17 17:12:22 +00:00
|
|
|
export type ButtonColor = NonNullable<InferredVariantProps["color"]>;
|
2022-07-23 00:39:50 +00:00
|
|
|
export type ButtonBaseProps = {
|
|
|
|
/** Action that happens when the button is clicked */
|
|
|
|
onClick?: (event: React.MouseEvent<HTMLElement, MouseEvent>) => void;
|
|
|
|
/**Left aligned icon*/
|
2022-12-18 10:44:44 +00:00
|
|
|
StartIcon?: SVGComponent | React.ElementType;
|
2022-07-23 00:39:50 +00:00
|
|
|
/**Right aligned icon */
|
2022-12-18 10:44:44 +00:00
|
|
|
EndIcon?: SVGComponent;
|
2022-07-23 00:39:50 +00:00
|
|
|
shallow?: boolean;
|
|
|
|
/**Tool tip used when icon size is set to small */
|
|
|
|
tooltip?: string;
|
2023-07-06 09:52:47 +00:00
|
|
|
disabled?: boolean;
|
2022-08-03 16:01:29 +00:00
|
|
|
flex?: boolean;
|
2022-11-23 02:55:25 +00:00
|
|
|
} & Omit<InferredVariantProps, "color"> & {
|
2023-01-17 17:12:22 +00:00
|
|
|
color?: ButtonColor;
|
2022-11-23 02:55:25 +00:00
|
|
|
};
|
2022-11-22 17:07:55 +00:00
|
|
|
|
2022-07-23 00:39:50 +00:00
|
|
|
export type ButtonProps = ButtonBaseProps &
|
|
|
|
(
|
2022-09-02 19:00:41 +00:00
|
|
|
| (Omit<JSX.IntrinsicElements["a"], "href" | "onClick" | "ref"> & LinkProps)
|
|
|
|
| (Omit<JSX.IntrinsicElements["button"], "onClick" | "ref"> & { href?: never })
|
2022-07-23 00:39:50 +00:00
|
|
|
);
|
|
|
|
|
2023-07-27 13:47:48 +00:00
|
|
|
export const buttonClasses = cva(
|
2023-08-12 21:21:09 +00:00
|
|
|
"whitespace-nowrap inline-flex items-center text-sm font-medium relative rounded-md transition-colors disabled:cursor-not-allowed",
|
2022-11-22 17:07:55 +00:00
|
|
|
{
|
|
|
|
variants: {
|
2023-01-19 14:55:32 +00:00
|
|
|
variant: {
|
|
|
|
button: "",
|
|
|
|
icon: "flex justify-center",
|
2023-02-20 14:11:51 +00:00
|
|
|
fab: "rounded-full justify-center md:rounded-md radix-state-open:rotate-45 md:radix-state-open:rotate-0 transition-transform radix-state-open:shadown-none radix-state-open:ring-0 !shadow-none",
|
2023-01-19 14:55:32 +00:00
|
|
|
},
|
2022-11-22 17:07:55 +00:00
|
|
|
color: {
|
2023-07-06 09:52:47 +00:00
|
|
|
primary:
|
|
|
|
"bg-brand-default hover:bg-brand-emphasis focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-offset focus-visible:ring-brand-default text-brand disabled:bg-brand-subtle disabled:text-brand-subtle disabled:opacity-40 disabled:hover:bg-brand-subtle disabled:hover:text-brand-default disabled:hover:opacity-40",
|
|
|
|
secondary:
|
|
|
|
"text-emphasis border border-default bg-default hover:bg-muted hover:border-emphasis focus-visible:bg-subtle focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-offset focus-visible:ring-empthasis disabled:border-subtle disabled:bg-opacity-30 disabled:text-muted disabled:hover:bg-opacity-30 disabled:hover:text-muted disabled:hover:border-subtle disabled:hover:bg-default",
|
|
|
|
minimal:
|
|
|
|
"text-emphasis hover:bg-subtle focus-visible:bg-subtle focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-offset focus-visible:ring-empthasis disabled:border-subtle disabled:bg-opacity-30 disabled:text-muted disabled:hover:bg-transparent disabled:hover:text-muted disabled:hover:border-subtle",
|
|
|
|
destructive:
|
2023-08-31 16:18:27 +00:00
|
|
|
"border border-default text-emphasis hover:text-red-700 dark:hover:text-red-100 focus-visible:text-red-700 hover:border-red-100 focus-visible:border-red-100 hover:bg-error focus-visible:bg-error focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-offset focus-visible:ring-red-700 disabled:bg-red-100 disabled:border-red-200 disabled:text-red-700 disabled:hover:border-red-200 disabled:opacity-40",
|
2022-11-22 17:07:55 +00:00
|
|
|
},
|
|
|
|
size: {
|
2022-11-23 02:55:25 +00:00
|
|
|
sm: "px-3 py-2 leading-4 rounded-sm" /** For backwards compatibility */,
|
|
|
|
base: "h-9 px-4 py-2.5 ",
|
2022-11-22 17:07:55 +00:00
|
|
|
lg: "h-[36px] px-4 py-2.5 ",
|
|
|
|
},
|
|
|
|
loading: {
|
|
|
|
true: "cursor-wait",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
compoundVariants: [
|
|
|
|
// Primary variants
|
|
|
|
{
|
|
|
|
loading: true,
|
|
|
|
color: "primary",
|
2023-04-08 16:50:17 +00:00
|
|
|
className: "bg-brand-subtle text-brand-subtle",
|
2022-11-22 17:07:55 +00:00
|
|
|
},
|
|
|
|
// Secondary variants
|
|
|
|
{
|
|
|
|
loading: true,
|
|
|
|
color: "secondary",
|
2023-04-05 18:14:46 +00:00
|
|
|
className: "bg-subtle text-emphasis/80",
|
2022-11-22 17:07:55 +00:00
|
|
|
},
|
|
|
|
// Minimal variants
|
|
|
|
{
|
|
|
|
loading: true,
|
|
|
|
color: "minimal",
|
2023-04-05 18:14:46 +00:00
|
|
|
className: "bg-subtle text-emphasis/30",
|
2022-11-22 17:07:55 +00:00
|
|
|
},
|
|
|
|
// Destructive variants
|
|
|
|
{
|
|
|
|
loading: true,
|
|
|
|
color: "destructive",
|
|
|
|
className:
|
2023-04-05 18:14:46 +00:00
|
|
|
"text-red-700/30 dark:text-red-700/30 hover:text-red-700/30 border border-default text-emphasis",
|
2022-11-22 17:07:55 +00:00
|
|
|
},
|
2023-01-19 14:55:32 +00:00
|
|
|
{
|
|
|
|
variant: "icon",
|
|
|
|
size: "base",
|
|
|
|
className: "min-h-[36px] min-w-[36px] !p-2",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "icon",
|
|
|
|
size: "sm",
|
|
|
|
className: "h-6 w-6 !p-1",
|
|
|
|
},
|
2023-01-21 16:14:08 +00:00
|
|
|
{
|
|
|
|
variant: "fab",
|
|
|
|
size: "base",
|
2023-02-02 08:24:31 +00:00
|
|
|
className: "h-14 md:h-9 md:w-auto md:px-4 md:py-2.5",
|
2023-01-21 16:14:08 +00:00
|
|
|
},
|
2022-11-22 17:07:55 +00:00
|
|
|
],
|
|
|
|
defaultVariants: {
|
2023-01-19 14:55:32 +00:00
|
|
|
variant: "button",
|
2022-11-22 17:07:55 +00:00
|
|
|
color: "primary",
|
|
|
|
size: "base",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
);
|
2022-07-23 00:39:50 +00:00
|
|
|
|
|
|
|
export const Button = forwardRef<HTMLAnchorElement | HTMLButtonElement, ButtonProps>(function Button(
|
|
|
|
props: ButtonProps,
|
|
|
|
forwardedRef
|
|
|
|
) {
|
|
|
|
const {
|
|
|
|
loading = false,
|
2022-11-23 02:55:25 +00:00
|
|
|
color = "primary",
|
2022-11-22 17:07:55 +00:00
|
|
|
size,
|
2023-01-19 14:55:32 +00:00
|
|
|
variant = "button",
|
2022-09-15 12:10:41 +00:00
|
|
|
type = "button",
|
2022-07-23 00:39:50 +00:00
|
|
|
StartIcon,
|
|
|
|
EndIcon,
|
|
|
|
shallow,
|
|
|
|
// attributes propagated from `HTMLAnchorProps` or `HTMLButtonProps`
|
|
|
|
...passThroughProps
|
|
|
|
} = props;
|
|
|
|
// Buttons are **always** disabled if we're in a `loading` state
|
|
|
|
const disabled = props.disabled || loading;
|
|
|
|
// If pass an `href`-attr is passed it's `<a>`, otherwise it's a `<button />`
|
|
|
|
const isLink = typeof props.href !== "undefined";
|
|
|
|
const elementType = isLink ? "a" : "button";
|
|
|
|
const element = React.createElement(
|
|
|
|
elementType,
|
|
|
|
{
|
|
|
|
...passThroughProps,
|
|
|
|
disabled,
|
2022-09-15 12:10:41 +00:00
|
|
|
type: !isLink ? type : undefined,
|
2022-07-23 00:39:50 +00:00
|
|
|
ref: forwardedRef,
|
2023-07-06 09:52:47 +00:00
|
|
|
className: classNames(buttonClasses({ color, size, loading, variant }), props.className),
|
2022-07-23 00:39:50 +00:00
|
|
|
// if we click a disabled button, we prevent going through the click handler
|
|
|
|
onClick: disabled
|
|
|
|
? (e: React.MouseEvent<HTMLElement, MouseEvent>) => {
|
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
: props.onClick,
|
|
|
|
},
|
|
|
|
<>
|
|
|
|
{StartIcon && (
|
2023-01-02 08:28:12 +00:00
|
|
|
<>
|
2023-01-19 14:55:32 +00:00
|
|
|
{variant === "fab" ? (
|
2023-01-02 08:28:12 +00:00
|
|
|
<>
|
2023-06-22 22:25:37 +00:00
|
|
|
<StartIcon className="hidden h-4 w-4 stroke-[1.5px] ltr:-ml-1 ltr:mr-2 rtl:-mr-1 rtl:ml-2 md:inline-flex" />
|
2023-07-27 13:47:48 +00:00
|
|
|
<Plus data-testid="plus" className="inline h-6 w-6 md:hidden" />
|
2023-01-02 08:28:12 +00:00
|
|
|
</>
|
|
|
|
) : (
|
|
|
|
<StartIcon
|
|
|
|
className={classNames(
|
2023-01-19 14:55:32 +00:00
|
|
|
variant === "icon" && "h-4 w-4",
|
2023-02-24 14:21:31 +00:00
|
|
|
variant === "button" && "h-4 w-4 stroke-[1.5px] ltr:-ml-1 ltr:mr-2 rtl:-mr-1 rtl:ml-2"
|
2023-01-02 08:28:12 +00:00
|
|
|
)}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</>
|
2022-07-23 00:39:50 +00:00
|
|
|
)}
|
2023-02-02 08:24:31 +00:00
|
|
|
{variant === "fab" ? <span className="hidden md:inline">{props.children}</span> : props.children}
|
2022-07-23 00:39:50 +00:00
|
|
|
{loading && (
|
2023-06-22 22:25:37 +00:00
|
|
|
<div className="absolute left-1/2 top-1/2 -translate-x-1/2 -translate-y-1/2 transform">
|
2022-07-23 00:39:50 +00:00
|
|
|
<svg
|
2023-02-24 11:55:46 +00:00
|
|
|
className={classNames(
|
|
|
|
"mx-4 h-5 w-5 animate-spin",
|
2023-04-05 18:14:46 +00:00
|
|
|
color === "primary" ? "text-inverted" : "text-emphasis"
|
2023-02-24 11:55:46 +00:00
|
|
|
)}
|
2022-07-23 00:39:50 +00:00
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
fill="none"
|
|
|
|
viewBox="0 0 24 24">
|
|
|
|
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
|
|
|
|
<path
|
|
|
|
className="opacity-75"
|
|
|
|
fill="currentColor"
|
|
|
|
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
</div>
|
|
|
|
)}
|
2023-01-02 08:28:12 +00:00
|
|
|
{EndIcon && (
|
|
|
|
<>
|
2023-01-19 14:55:32 +00:00
|
|
|
{variant === "fab" ? (
|
2023-01-02 08:28:12 +00:00
|
|
|
<>
|
2023-06-22 22:25:37 +00:00
|
|
|
<EndIcon className="-mr-1 me-2 ms-2 hidden h-5 w-5 md:inline" />
|
2023-07-27 13:47:48 +00:00
|
|
|
<Plus data-testid="plus" className="inline h-6 w-6 md:hidden" />
|
2023-01-02 08:28:12 +00:00
|
|
|
</>
|
|
|
|
) : (
|
2023-01-19 14:55:32 +00:00
|
|
|
<EndIcon
|
|
|
|
className={classNames(
|
|
|
|
"inline-flex",
|
|
|
|
variant === "icon" && "h-4 w-4",
|
2023-06-22 22:25:37 +00:00
|
|
|
variant === "button" && "h-4 w-4 stroke-[1.5px] ltr:-mr-1 ltr:ml-2 rtl:-ml-1 rtl:mr-2"
|
2023-01-19 14:55:32 +00:00
|
|
|
)}
|
|
|
|
/>
|
2023-01-02 08:28:12 +00:00
|
|
|
)}
|
|
|
|
</>
|
|
|
|
)}
|
2022-07-23 00:39:50 +00:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
|
|
|
|
return props.href ? (
|
2023-07-27 13:47:48 +00:00
|
|
|
<Link data-testid="link-component" passHref href={props.href} shallow={shallow && shallow} legacyBehavior>
|
2022-07-23 00:39:50 +00:00
|
|
|
{element}
|
|
|
|
</Link>
|
|
|
|
) : (
|
2023-07-27 13:47:48 +00:00
|
|
|
<Wrapper data-testid="wrapper" tooltip={props.tooltip}>
|
|
|
|
{element}
|
|
|
|
</Wrapper>
|
2022-07-23 00:39:50 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
const Wrapper = ({ children, tooltip }: { tooltip?: string; children: React.ReactNode }) => {
|
|
|
|
if (!tooltip) {
|
|
|
|
return <>{children}</>;
|
|
|
|
}
|
|
|
|
|
2023-07-27 13:47:48 +00:00
|
|
|
return (
|
|
|
|
<Tooltip data-testid="tooltip" content={tooltip}>
|
|
|
|
{children}
|
|
|
|
</Tooltip>
|
|
|
|
);
|
2022-07-23 00:39:50 +00:00
|
|
|
};
|