import type { AnchorHTMLAttributes, ButtonHTMLAttributes, ReactNode, } from 'react'; import { forwardRef } from 'react'; import classNames from 'classnames'; import type { IconName } from '../icon'; import { Icon } from '../icon'; import { includesLeftPadding, includesRightPadding, includesBorderWidth, includesHeight, } from '../../utils/class-names'; import classnames from 'classnames'; interface CommonProps { children?: ReactNode; variant?: 'primary' | 'secondary' | 'trade' | 'accent' | 'inline-link'; className?: string; prependIconName?: IconName; appendIconName?: IconName; boxShadow?: boolean; } export interface ButtonProps extends ButtonHTMLAttributes, CommonProps {} export interface AnchorButtonProps extends AnchorHTMLAttributes, CommonProps {} export const getButtonClasses = ( className?: string, variant?: 'primary' | 'secondary' | 'trade' | 'accent' | 'inline-link', boxShadow?: boolean ) => { const paddingLeftProvided = includesLeftPadding(className); const paddingRightProvided = includesRightPadding(className); const borderWidthProvided = includesBorderWidth(className); const heightProvided = includesHeight(className); // Add classes into variables if there are multiple classes shared in multiple button styles const sharedClasses = 'inline-flex items-center justify-center box-border transition-[background-color] ease-linear duration-50 disabled:no-underline'; const commonButtonClasses = classnames( 'relative disabled:static', 'text-ui font-semibold focus-visible:outline-none border no-underline hover:no-underline', { 'shadow-none': !boxShadow, 'shadow-[3px_3px_0_0] focus-visible:shadow-vega-pink dark:focus-visible:shadow-vega-yellow active:top-[1px] active:left-[1px] active:shadow-[2px_2px_0_0]': boxShadow === undefined || boxShadow, } ); const commonDisabled = 'disabled:bg-black-10 dark:disabled:bg-white-10 disabled:text-black-60 dark:disabled:text-white-60 disabled:border-black-25 dark:disabled:border-white-25 disabled:shadow-none dark:disabled:shadow-none'; const inlineTextColour = 'text-black-95 dark:text-white-95 hover:text-black hover:dark:text-white active:text-black dark:active:text-vega-yellow'; const standardButtonPaddingLeft = `${ paddingLeftProvided ? paddingLeftProvided : 'pl-28' }`; const standardButtonPaddingRight = `${ paddingRightProvided ? paddingRightProvided : 'pr-28' }`; const inlineButtonPaddingLeft = `${ paddingLeftProvided ? paddingLeftProvided : 'pl-4' }`; const inlineButtonPaddingRight = `${ paddingRightProvided ? paddingRightProvided : 'pr-4' }`; const standardButtonBorderWidth = `${ borderWidthProvided ? borderWidthProvided : 'border' }`; const buttonHeight = `${heightProvided ? heightProvided : 'h-28'}`; const primaryClasses = [ sharedClasses, commonButtonClasses, commonDisabled, standardButtonPaddingLeft, standardButtonPaddingRight, standardButtonBorderWidth, buttonHeight, 'bg-black dark:bg-white hover:bg-black-80 dark:hover:bg-white-90 active:bg-black-80 dark:active:bg-white-90', 'border-white dark:border-black shadow-black active:shadow-black dark:shadow-white-80 dark:active:shadow-white', 'text-white dark:text-black', ]; const secondaryClasses = [ sharedClasses, commonButtonClasses, commonDisabled, standardButtonPaddingLeft, standardButtonPaddingRight, standardButtonBorderWidth, buttonHeight, 'bg-white dark:bg-black hover:bg-black-25 dark:hover:bg-white-25', 'border-black dark:border-white shadow-black dark:shadow-white', 'text-black dark:text-white', ]; const tradeClasses = [ sharedClasses, commonButtonClasses, commonDisabled, standardButtonPaddingLeft, standardButtonPaddingRight, standardButtonBorderWidth, buttonHeight, 'bg-vega-green hover:bg-vega-green-medium', 'border-black disabled:shadow-none dark:disabled:shadow-none shadow-black dark:shadow-white', 'text-black', ]; const accentClasses = [ sharedClasses, commonDisabled, standardButtonPaddingLeft, standardButtonPaddingRight, standardButtonBorderWidth, buttonHeight, 'bg-vega-yellow dark:bg-vega-yellow hover:bg-vega-yellow-dark dark:hover:bg-vega-yellow-dark active:bg-white dark:active:bg-black', 'uppercase text-black dark:text-black hover:text-white dark:hover:text-white active:text-black dark:active:text-white', 'border-transparent dark:border-transparent', ]; const inlineLinkClasses = [ sharedClasses, inlineButtonPaddingLeft, inlineButtonPaddingRight, buttonHeight, inlineTextColour, 'underline hover:underline hover:text-black-60 dark:hover:text-white-80', 'border-none', ]; let variantClasses: string[]; switch (variant) { case 'primary': variantClasses = primaryClasses; break; case 'secondary': variantClasses = secondaryClasses; break; case 'trade': variantClasses = tradeClasses; break; case 'accent': variantClasses = accentClasses; break; case 'inline-link': variantClasses = inlineLinkClasses; break; default: variantClasses = ['']; } return classNames(...variantClasses, className); }; export const getButtonContent = ( children: ReactNode, prependIconName?: IconName, appendIconName?: IconName ) => { const iconName = prependIconName || appendIconName; if (iconName === undefined) { return children; } const iconClassName = classNames(['fill-current'], { 'mr-8': prependIconName, 'ml-8': appendIconName, }); const icon = ; return ( <> {prependIconName && icon} {children} {appendIconName && icon} ); }; export const Button = forwardRef( ( { variant = 'primary', type = 'button', children, className, prependIconName, appendIconName, boxShadow, ...props }, ref ) => { return ( ); } ); export const AnchorButton = forwardRef( ( { variant = 'primary', children, className, prependIconName, appendIconName, boxShadow, ...props }, ref ) => { return ( {getButtonContent(children, prependIconName, appendIconName)} ); } );