UNPKG

@nlabs/gothamjs

Version:
112 lines (111 loc) 13.1 kB
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime"; import { cn } from '@nlabs/utils'; import { forwardRef } from 'react'; import { useTranslation } from '../../i18n/index.js'; import { getBackgroundClasses, getBorderClasses, getTextClasses } from '../../utils/colorUtils.js'; export const Button = /*#__PURE__*/ forwardRef(({ children, className, color = 'primary', disabled, hasNotification = false, hasShadow = false, icon, isLoading = false, label = '', onClick = ()=>{}, size = 'md', type = 'button', variant, ...restBtnProps }, ref)=>{ const { t } = useTranslation(); const classes = []; if (variant) { classes.push(...[ 'cursor-pointer', 'flex', 'focus-visible:outline', 'focus-visible:outline-2', 'focus-visible:outline-offset-2', 'focus-visible:outline-secondary', 'items-center', 'justify-center', 'leading-6', 'relative' ]); switch(variant){ case 'outlined': classes.push('bg-transparent', 'border-1', 'font-semibold', getBorderClasses(color, { hasFocus: true, hasHover: true }), getTextClasses(color, { hasFocus: true, hasHover: true })); break; case 'text': classes.push('bg-transparent', 'font-semibold', getTextClasses(color, { hasFocus: true, hasHover: true })); break; case 'contained': default: classes.push(getBackgroundClasses(color, { hasFocus: true, hasHover: true }), 'font-medium', 'text-white', hasShadow ? 'shadow-sm' : ''); break; } switch(size){ case 'sm': classes.push('px-4', 'py-0.5', 'rounded', 'text-sm'); break; case 'lg': classes.push('px-8', 'py-4', 'rounded-lg', 'text-lg'); break; case 'md': default: classes.push('px-6', 'py-2', 'rounded-md', 'text-md'); } } let buttonIcon = icon; if (isLoading) { buttonIcon = /*#__PURE__*/ _jsxs("svg", { className: "mr-3 -ml-1 size-5 animate-spin text-white", xmlns: "http://www.w3.org/2000/svg", fill: "none", viewBox: "0 0 24 24", children: [ /*#__PURE__*/ _jsx("circle", { className: "opacity-25", cx: "12", cy: "12", r: "10", stroke: "currentColor", "stroke-width": "4" }), /*#__PURE__*/ _jsx("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" }) ] }); } let buttonNotification = null; if (hasNotification) { buttonNotification = /*#__PURE__*/ _jsxs("span", { className: "absolute top-0 right-0 -mt-1 -mr-1 flex size-3", children: [ /*#__PURE__*/ _jsx("span", { className: "absolute inline-flex h-full w-full animate-ping rounded-full bg-sky-400 opacity-75" }), /*#__PURE__*/ _jsx("span", { className: "relative inline-flex size-3 rounded-full bg-sky-500" }) ] }); } return /*#__PURE__*/ _jsxs("button", { className: cn(...classes, className), "data-testid": `button-${label || children}`, disabled: isLoading || disabled, onClick: onClick, ref: ref, type: type, ...restBtnProps, children: [ buttonNotification, buttonIcon, children ? children : t(label) ] }); }); //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/gothamjs/src/components/Button/Button.tsx"],"sourcesContent":["import {cn} from '@nlabs/utils';\nimport {forwardRef} from 'react';\n\nimport {useTranslation} from '../../i18n/index.js';\nimport {getBackgroundClasses, getBorderClasses, getTextClasses} from '../../utils/colorUtils.js';\n\nimport type {ReactNode} from 'react';\nimport type {GothamColor} from '../../utils/colorUtils.js';\nimport type {GothamSize} from '../../utils/sizeUtils.js';\n\nexport type ButtonType = 'button' | 'reset' | 'submit';\nexport type ButtonVariant = 'text' | 'contained' | 'outlined';\n\nexport interface ButtonProps {\n\treadonly children?: ReactNode;\n\treadonly className?: string;\n\treadonly color?: GothamColor;\n\treadonly disabled?: boolean;\n\treadonly hasNotification?: boolean;\n\treadonly hasShadow?: boolean;\n\treadonly icon?: ReactNode;\n\treadonly isLoading?: boolean;\n\treadonly label?: string;\n\treadonly onClick?: (event?: unknown) => void;\n\treadonly size?: GothamSize;\n  readonly tabIndex?: number;\n  readonly type?: ButtonType;\n  readonly variant?: ButtonVariant;\n}\n\nexport const Button = forwardRef<HTMLButtonElement, ButtonProps>(({\n  children,\n  className,\n  color = 'primary',\n  disabled,\n  hasNotification = false,\n  hasShadow = false,\n  icon,\n  isLoading = false,\n  label = '',\n  onClick = () => {},\n  size = 'md',\n  type = 'button',\n  variant,\n  ...restBtnProps\n}, ref) => {\n  const {t} = useTranslation();\n  const classes: string[] = [];\n\n  if(variant) {\n    classes.push(...[\n      'cursor-pointer',\n      'flex',\n      'focus-visible:outline',\n      'focus-visible:outline-2',\n      'focus-visible:outline-offset-2',\n      'focus-visible:outline-secondary',\n      'items-center',\n      'justify-center',\n      'leading-6',\n      'relative'\n    ]);\n\n    switch(variant) {\n      case 'outlined':\n        classes.push(\n          'bg-transparent',\n          'border-1',\n          'font-semibold',\n          getBorderClasses(color, {hasFocus: true, hasHover: true}),\n          getTextClasses(color, {hasFocus: true, hasHover: true})\n        );\n        break;\n      case 'text':\n        classes.push(\n          'bg-transparent',\n          'font-semibold',\n          getTextClasses(color, {hasFocus: true, hasHover: true})\n        );\n        break;\n      case 'contained':\n      default:\n        classes.push(\n          getBackgroundClasses(color, {hasFocus: true, hasHover: true}),\n          'font-medium',\n          'text-white',\n          hasShadow ? 'shadow-sm' : ''\n        );\n        break;\n    }\n\n    switch(size) {\n      case 'sm':\n        classes.push(\n          'px-4',\n          'py-0.5',\n          'rounded',\n          'text-sm'\n        );\n        break;\n      case 'lg':\n        classes.push(\n          'px-8',\n          'py-4',\n          'rounded-lg',\n          'text-lg'\n        );\n        break;\n      case 'md':\n      default:\n        classes.push(\n          'px-6',\n          'py-2',\n          'rounded-md',\n          'text-md'\n        );\n    }\n  }\n\n  let buttonIcon = icon;\n\n  if(isLoading) {\n    buttonIcon = (\n      <svg className=\"mr-3 -ml-1 size-5 animate-spin text-white\" xmlns=\"http://www.w3.org/2000/svg\" fill=\"none\" viewBox=\"0 0 24 24\">\n        <circle className=\"opacity-25\" cx=\"12\" cy=\"12\" r=\"10\" stroke=\"currentColor\" stroke-width=\"4\"></circle>\n        <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\"></path>\n      </svg>\n    );\n  }\n\n  let buttonNotification: ReactNode | null = null;\n\n  if(hasNotification) {\n    buttonNotification = (\n      <span className=\"absolute top-0 right-0 -mt-1 -mr-1 flex size-3\">\n        <span className=\"absolute inline-flex h-full w-full animate-ping rounded-full bg-sky-400 opacity-75\"></span>\n        <span className=\"relative inline-flex size-3 rounded-full bg-sky-500\"></span>\n      </span>\n    );\n  }\n\n  return (\n    <button\n      className={cn(...classes, className)}\n      data-testid={`button-${label || children}`}\n      disabled={isLoading || disabled}\n      onClick={onClick}\n      ref={ref}\n      type={type}\n      {...restBtnProps}\n    >\n      {buttonNotification}\n      {buttonIcon}\n      {children ? children : t(label)}\n    </button>\n  );\n});\n"],"names":["cn","forwardRef","useTranslation","getBackgroundClasses","getBorderClasses","getTextClasses","Button","children","className","color","disabled","hasNotification","hasShadow","icon","isLoading","label","onClick","size","type","variant","restBtnProps","ref","t","classes","push","hasFocus","hasHover","buttonIcon","svg","xmlns","fill","viewBox","circle","cx","cy","r","stroke","stroke-width","path","d","buttonNotification","span","button","data-testid"],"mappings":";AAAA,SAAQA,EAAE,QAAO,eAAe;AAChC,SAAQC,UAAU,QAAO,QAAQ;AAEjC,SAAQC,cAAc,QAAO,sBAAsB;AACnD,SAAQC,oBAAoB,EAAEC,gBAAgB,EAAEC,cAAc,QAAO,4BAA4B;AA0BjG,OAAO,MAAMC,uBAASL,WAA2C,CAAC,EAChEM,QAAQ,EACRC,SAAS,EACTC,QAAQ,SAAS,EACjBC,QAAQ,EACRC,kBAAkB,KAAK,EACvBC,YAAY,KAAK,EACjBC,IAAI,EACJC,YAAY,KAAK,EACjBC,QAAQ,EAAE,EACVC,UAAU,KAAO,CAAC,EAClBC,OAAO,IAAI,EACXC,OAAO,QAAQ,EACfC,OAAO,EACP,GAAGC,cACJ,EAAEC;IACD,MAAM,EAACC,CAAC,EAAC,GAAGpB;IACZ,MAAMqB,UAAoB,EAAE;IAE5B,IAAGJ,SAAS;QACVI,QAAQC,IAAI,IAAI;YACd;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;SACD;QAED,OAAOL;YACL,KAAK;gBACHI,QAAQC,IAAI,CACV,kBACA,YACA,iBACApB,iBAAiBK,OAAO;oBAACgB,UAAU;oBAAMC,UAAU;gBAAI,IACvDrB,eAAeI,OAAO;oBAACgB,UAAU;oBAAMC,UAAU;gBAAI;gBAEvD;YACF,KAAK;gBACHH,QAAQC,IAAI,CACV,kBACA,iBACAnB,eAAeI,OAAO;oBAACgB,UAAU;oBAAMC,UAAU;gBAAI;gBAEvD;YACF,KAAK;YACL;gBACEH,QAAQC,IAAI,CACVrB,qBAAqBM,OAAO;oBAACgB,UAAU;oBAAMC,UAAU;gBAAI,IAC3D,eACA,cACAd,YAAY,cAAc;gBAE5B;QACJ;QAEA,OAAOK;YACL,KAAK;gBACHM,QAAQC,IAAI,CACV,QACA,UACA,WACA;gBAEF;YACF,KAAK;gBACHD,QAAQC,IAAI,CACV,QACA,QACA,cACA;gBAEF;YACF,KAAK;YACL;gBACED,QAAQC,IAAI,CACV,QACA,QACA,cACA;QAEN;IACF;IAEA,IAAIG,aAAad;IAEjB,IAAGC,WAAW;QACZa,2BACE,MAACC;YAAIpB,WAAU;YAA4CqB,OAAM;YAA6BC,MAAK;YAAOC,SAAQ;;8BAChH,KAACC;oBAAOxB,WAAU;oBAAayB,IAAG;oBAAKC,IAAG;oBAAKC,GAAE;oBAAKC,QAAO;oBAAeC,gBAAa;;8BACzF,KAACC;oBAAK9B,WAAU;oBAAasB,MAAK;oBAAeS,GAAE;;;;IAGzD;IAEA,IAAIC,qBAAuC;IAE3C,IAAG7B,iBAAiB;QAClB6B,mCACE,MAACC;YAAKjC,WAAU;;8BACd,KAACiC;oBAAKjC,WAAU;;8BAChB,KAACiC;oBAAKjC,WAAU;;;;IAGtB;IAEA,qBACE,MAACkC;QACClC,WAAWR,MAAMuB,SAASf;QAC1BmC,eAAa,CAAC,OAAO,EAAE5B,SAASR,UAAU;QAC1CG,UAAUI,aAAaJ;QACvBM,SAASA;QACTK,KAAKA;QACLH,MAAMA;QACL,GAAGE,YAAY;;YAEfoB;YACAb;YACApB,WAAWA,WAAWe,EAAEP;;;AAG/B,GAAG"}