@ultraviolet/ui 2.1.4 → 2.1.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -100,7 +100,7 @@ const StyledFilledButton = /* @__PURE__ */ _styled__default.default("button", pr
100
100
  color:
101
101
  ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].textHover};
102
102
  }
103
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAyGqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
103
+ `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAyGqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  'aria-keyshortcuts'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-keyshortcuts={ariaKeyshortcuts}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
104
104
  const StyledOutlinedButton = /* @__PURE__ */ _styled__default.default("button", process.env.NODE_ENV === "production" ? {
105
105
  shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
106
106
  target: "e112qvla1"
@@ -132,7 +132,7 @@ const StyledOutlinedButton = /* @__PURE__ */ _styled__default.default("button",
132
132
  border: 1px solid ${!isMonochrome(sentiment) ? theme.colors[sentiment][sentiment === "neutral" ? "borderStrongHover" : "borderHover"] : theme.colors.other.monochrome[sentiment].borderHover};
133
133
 
134
134
  }
135
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AA0IqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
135
+ `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AA0IqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  'aria-keyshortcuts'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-keyshortcuts={ariaKeyshortcuts}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
136
136
  const StyledGhostButton = /* @__PURE__ */ _styled__default.default("button", process.env.NODE_ENV === "production" ? {
137
137
  shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
138
138
  target: "e112qvla0"
@@ -157,7 +157,7 @@ const StyledGhostButton = /* @__PURE__ */ _styled__default.default("button", pro
157
157
  color:
158
158
  ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].textHover};
159
159
  }
160
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAiMqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
160
+ `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAiMqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  'aria-keyshortcuts'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-keyshortcuts={ariaKeyshortcuts}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
161
161
  const VARIANTS_COMPONENTS = {
162
162
  filled: {
163
163
  button: StyledFilledButton,
@@ -216,6 +216,7 @@ const Button = react.forwardRef(({
216
216
  "aria-disabled": ariaDisabled,
217
217
  "aria-pressed": ariaPressed,
218
218
  "aria-roledescription": ariaRoledescription,
219
+ "aria-keyshortcuts": ariaKeyshortcuts,
219
220
  href,
220
221
  download,
221
222
  target,
@@ -243,7 +244,7 @@ const Button = react.forwardRef(({
243
244
  ] });
244
245
  if (href && !computeIsDisabled) {
245
246
  const Component2 = VARIANTS_COMPONENTS[variant].link;
246
- return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsx(Component2, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-describedby": ariaDescribedby, "aria-disabled": ariaDisabled ?? disabled, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, "aria-pressed": ariaPressed, "aria-roledescription": ariaRoledescription, autoFocus, className, "data-testid": dataTestId, disabled: false, download, fullWidth, href, onClick, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, ref, role, sentiment, size, tabIndex, target, type, children: content }) });
247
+ return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsx(Component2, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-describedby": ariaDescribedby, "aria-disabled": ariaDisabled ?? disabled, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-keyshortcuts": ariaKeyshortcuts, "aria-label": ariaLabel, "aria-pressed": ariaPressed, "aria-roledescription": ariaRoledescription, autoFocus, className, "data-testid": dataTestId, disabled: false, download, fullWidth, href, onClick, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, ref, role, sentiment, size, tabIndex, target, type, children: content }) });
247
248
  }
248
249
  const Component = VARIANTS_COMPONENTS[variant].button;
249
250
  return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsx(Component, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, autoFocus, className, "data-testid": dataTestId, disabled: computeIsDisabled, fullWidth, name, onClick, onKeyDown, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, onPointerDown, ref, role, sentiment, size, tabIndex, type, children: content }) });
@@ -59,6 +59,7 @@ type CommonProps = {
59
59
  'aria-disabled'?: boolean;
60
60
  'aria-pressed'?: boolean;
61
61
  'aria-roledescription'?: string;
62
+ 'aria-keyshortcuts'?: string;
62
63
  onClick?: MouseEventHandler<HTMLElement>;
63
64
  tooltip?: string;
64
65
  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex'];
@@ -96,7 +96,7 @@ const StyledFilledButton = /* @__PURE__ */ _styled("button", process.env.NODE_EN
96
96
  color:
97
97
  ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].textHover};
98
98
  }
99
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAyGqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
99
+ `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAyGqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  'aria-keyshortcuts'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-keyshortcuts={ariaKeyshortcuts}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
100
100
  const StyledOutlinedButton = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
101
101
  shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
102
102
  target: "e112qvla1"
@@ -128,7 +128,7 @@ const StyledOutlinedButton = /* @__PURE__ */ _styled("button", process.env.NODE_
128
128
  border: 1px solid ${!isMonochrome(sentiment) ? theme.colors[sentiment][sentiment === "neutral" ? "borderStrongHover" : "borderHover"] : theme.colors.other.monochrome[sentiment].borderHover};
129
129
 
130
130
  }
131
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AA0IqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
131
+ `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AA0IqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  'aria-keyshortcuts'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-keyshortcuts={ariaKeyshortcuts}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
132
132
  const StyledGhostButton = /* @__PURE__ */ _styled("button", process.env.NODE_ENV === "production" ? {
133
133
  shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
134
134
  target: "e112qvla0"
@@ -153,7 +153,7 @@ const StyledGhostButton = /* @__PURE__ */ _styled("button", process.env.NODE_ENV
153
153
  color:
154
154
  ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].textHover};
155
155
  }
156
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAiMqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
156
+ `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAiMqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  'aria-keyshortcuts'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-keyshortcuts={ariaKeyshortcuts}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
157
157
  const VARIANTS_COMPONENTS = {
158
158
  filled: {
159
159
  button: StyledFilledButton,
@@ -212,6 +212,7 @@ const Button = forwardRef(({
212
212
  "aria-disabled": ariaDisabled,
213
213
  "aria-pressed": ariaPressed,
214
214
  "aria-roledescription": ariaRoledescription,
215
+ "aria-keyshortcuts": ariaKeyshortcuts,
215
216
  href,
216
217
  download,
217
218
  target,
@@ -239,7 +240,7 @@ const Button = forwardRef(({
239
240
  ] });
240
241
  if (href && !computeIsDisabled) {
241
242
  const Component2 = VARIANTS_COMPONENTS[variant].link;
242
- return /* @__PURE__ */ jsx(Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsx(Component2, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-describedby": ariaDescribedby, "aria-disabled": ariaDisabled ?? disabled, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, "aria-pressed": ariaPressed, "aria-roledescription": ariaRoledescription, autoFocus, className, "data-testid": dataTestId, disabled: false, download, fullWidth, href, onClick, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, ref, role, sentiment, size, tabIndex, target, type, children: content }) });
243
+ return /* @__PURE__ */ jsx(Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsx(Component2, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-describedby": ariaDescribedby, "aria-disabled": ariaDisabled ?? disabled, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-keyshortcuts": ariaKeyshortcuts, "aria-label": ariaLabel, "aria-pressed": ariaPressed, "aria-roledescription": ariaRoledescription, autoFocus, className, "data-testid": dataTestId, disabled: false, download, fullWidth, href, onClick, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, ref, role, sentiment, size, tabIndex, target, type, children: content }) });
243
244
  }
244
245
  const Component = VARIANTS_COMPONENTS[variant].button;
245
246
  return /* @__PURE__ */ jsx(Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsx(Component, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, autoFocus, className, "data-testid": dataTestId, disabled: computeIsDisabled, fullWidth, name, onClick, onKeyDown, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, onPointerDown, ref, role, sentiment, size, tabIndex, type, children: content }) });
@@ -34,6 +34,7 @@ export declare const Dialog: (({ ariaLabel, className, children, "data-testid":
34
34
  'aria-disabled'?: boolean;
35
35
  'aria-pressed'?: boolean;
36
36
  'aria-roledescription'?: string;
37
+ 'aria-keyshortcuts'?: string;
37
38
  onClick?: import("react").MouseEventHandler<HTMLElement>;
38
39
  tooltip?: string;
39
40
  tabIndex?: import("react").ButtonHTMLAttributes<HTMLButtonElement>["tabIndex"];
@@ -17,7 +17,7 @@ const StyledArrowLeftIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowL
17
17
  label: "StyledArrowLeftIcon"
18
18
  })("margin-right:", ({
19
19
  theme
20
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
20
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
21
21
  const StyledArrowRightIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowRightIcon, process.env.NODE_ENV === "production" ? {
22
22
  target: "e1afnb7a2"
23
23
  } : {
@@ -25,7 +25,7 @@ const StyledArrowRightIcon = /* @__PURE__ */ _styled__default.default(Icon.Arrow
25
25
  label: "StyledArrowRightIcon"
26
26
  })("margin-left:", ({
27
27
  theme
28
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAyBmD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
28
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAyBmD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
29
29
  const StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(Icon.OpenInNewIcon, process.env.NODE_ENV === "production" ? {
30
30
  target: "e1afnb7a4"
31
31
  } : {
@@ -48,7 +48,7 @@ const StyledExternalIconContainer = /* @__PURE__ */ _styled__default.default("sp
48
48
  label: "StyledExternalIconContainer"
49
49
  })("display:inline-flex;padding-bottom:", ({
50
50
  theme
51
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAkE+C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
51
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAmE+C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
52
52
  const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NODE_ENV === "production" ? {
53
53
  shouldForwardProp: (prop) => !["sentiment", "iconPosition", "as", "oneLine"].includes(prop),
54
54
  target: "e1afnb7a0"
@@ -116,7 +116,7 @@ const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NOD
116
116
  }, ";}}&[data-variant='inline']{text-decoration:underline;text-decoration-thickness:1px;}&:hover::after,&:focus::after{background-color:", ({
117
117
  theme,
118
118
  sentiment
119
- }) => theme.colors[sentiment]?.text ?? theme.colors.neutral.text, ";}&:active{text-decoration-thickness:2px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAgFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
119
+ }) => theme.colors[sentiment]?.text ?? theme.colors.neutral.text, ";}&:active{text-decoration-thickness:2px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAiFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
120
120
  const Link = react.forwardRef(({
121
121
  children,
122
122
  href,
@@ -131,6 +131,7 @@ const Link = react.forwardRef(({
131
131
  onClick,
132
132
  "aria-label": ariaLabel,
133
133
  "aria-current": ariaCurrent,
134
+ "aria-keyshortcuts": ariaKeyshortcuts,
134
135
  oneLine = false,
135
136
  "data-testid": dataTestId,
136
137
  variant = "standalone"
@@ -159,7 +160,7 @@ const Link = react.forwardRef(({
159
160
  setIsTruncated(offsetWidth < scrollWidth);
160
161
  }
161
162
  }, [oneLine, ref, usedRef]);
162
- return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { text: oneLine && isTruncated ? finalStringChildren : "", children: /* @__PURE__ */ jsxRuntime.jsxs(StyledLink, { "aria-current": ariaCurrent, "aria-label": ariaLabel, className, "data-testid": dataTestId, "data-variant": variant, download, href, iconPosition, onClick, oneLine, prominence, ref: usedRef, rel: computedRel, sentiment, target, variant: textVariant, children: [
163
+ return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { text: oneLine && isTruncated ? finalStringChildren : "", children: /* @__PURE__ */ jsxRuntime.jsxs(StyledLink, { "aria-current": ariaCurrent, "aria-keyshortcuts": ariaKeyshortcuts, "aria-label": ariaLabel, className, "data-testid": dataTestId, "data-variant": variant, download, href, iconPosition, onClick, oneLine, prominence, ref: usedRef, rel: computedRel, sentiment, target, variant: textVariant, children: [
163
164
  !isBlank && iconPosition === "left" ? /* @__PURE__ */ jsxRuntime.jsx(StyledArrowLeftIcon, { size: ICON_SIZE }) : null,
164
165
  children,
165
166
  isBlank ? /* @__PURE__ */ jsxRuntime.jsx(StyledExternalIconContainer, { children: /* @__PURE__ */ jsxRuntime.jsx(StyledOpenInNewIcon, { size: BLANK_TARGET_ICON_SIZE }) }) : null,
@@ -22,6 +22,7 @@ type LinkProps = {
22
22
  onClick?: MouseEventHandler<HTMLAnchorElement>;
23
23
  'aria-label'?: string;
24
24
  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current'];
25
+ 'aria-keyshortcuts'?: string;
25
26
  oneLine?: boolean;
26
27
  'data-testid'?: string;
27
28
  variant?: 'inline' | 'standalone';
@@ -13,7 +13,7 @@ const StyledArrowLeftIcon = /* @__PURE__ */ _styled(ArrowLeftIcon, process.env.N
13
13
  label: "StyledArrowLeftIcon"
14
14
  })("margin-right:", ({
15
15
  theme
16
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
16
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
17
17
  const StyledArrowRightIcon = /* @__PURE__ */ _styled(ArrowRightIcon, process.env.NODE_ENV === "production" ? {
18
18
  target: "e1afnb7a2"
19
19
  } : {
@@ -21,7 +21,7 @@ const StyledArrowRightIcon = /* @__PURE__ */ _styled(ArrowRightIcon, process.env
21
21
  label: "StyledArrowRightIcon"
22
22
  })("margin-left:", ({
23
23
  theme
24
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAyBmD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
24
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAyBmD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
25
25
  const StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon, process.env.NODE_ENV === "production" ? {
26
26
  target: "e1afnb7a4"
27
27
  } : {
@@ -44,7 +44,7 @@ const StyledExternalIconContainer = /* @__PURE__ */ _styled("span", process.env.
44
44
  label: "StyledExternalIconContainer"
45
45
  })("display:inline-flex;padding-bottom:", ({
46
46
  theme
47
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAkE+C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
47
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAmE+C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
48
48
  const StyledLink = /* @__PURE__ */ _styled("a", process.env.NODE_ENV === "production" ? {
49
49
  shouldForwardProp: (prop) => !["sentiment", "iconPosition", "as", "oneLine"].includes(prop),
50
50
  target: "e1afnb7a0"
@@ -112,7 +112,7 @@ const StyledLink = /* @__PURE__ */ _styled("a", process.env.NODE_ENV === "produc
112
112
  }, ";}}&[data-variant='inline']{text-decoration:underline;text-decoration-thickness:1px;}&:hover::after,&:focus::after{background-color:", ({
113
113
  theme,
114
114
  sentiment
115
- }) => theme.colors[sentiment]?.text ?? theme.colors.neutral.text, ";}&:active{text-decoration-thickness:2px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAgFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
115
+ }) => theme.colors[sentiment]?.text ?? theme.colors.neutral.text, ";}&:active{text-decoration-thickness:2px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAiFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  'aria-keyshortcuts'?: string\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-keyshortcuts': ariaKeyshortcuts,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-keyshortcuts={ariaKeyshortcuts}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
116
116
  const Link = forwardRef(({
117
117
  children,
118
118
  href,
@@ -127,6 +127,7 @@ const Link = forwardRef(({
127
127
  onClick,
128
128
  "aria-label": ariaLabel,
129
129
  "aria-current": ariaCurrent,
130
+ "aria-keyshortcuts": ariaKeyshortcuts,
130
131
  oneLine = false,
131
132
  "data-testid": dataTestId,
132
133
  variant = "standalone"
@@ -155,7 +156,7 @@ const Link = forwardRef(({
155
156
  setIsTruncated(offsetWidth < scrollWidth);
156
157
  }
157
158
  }, [oneLine, ref, usedRef]);
158
- return /* @__PURE__ */ jsx(Tooltip, { text: oneLine && isTruncated ? finalStringChildren : "", children: /* @__PURE__ */ jsxs(StyledLink, { "aria-current": ariaCurrent, "aria-label": ariaLabel, className, "data-testid": dataTestId, "data-variant": variant, download, href, iconPosition, onClick, oneLine, prominence, ref: usedRef, rel: computedRel, sentiment, target, variant: textVariant, children: [
159
+ return /* @__PURE__ */ jsx(Tooltip, { text: oneLine && isTruncated ? finalStringChildren : "", children: /* @__PURE__ */ jsxs(StyledLink, { "aria-current": ariaCurrent, "aria-keyshortcuts": ariaKeyshortcuts, "aria-label": ariaLabel, className, "data-testid": dataTestId, "data-variant": variant, download, href, iconPosition, onClick, oneLine, prominence, ref: usedRef, rel: computedRel, sentiment, target, variant: textVariant, children: [
159
160
  !isBlank && iconPosition === "left" ? /* @__PURE__ */ jsx(StyledArrowLeftIcon, { size: ICON_SIZE }) : null,
160
161
  children,
161
162
  isBlank ? /* @__PURE__ */ jsx(StyledExternalIconContainer, { children: /* @__PURE__ */ jsx(StyledOpenInNewIcon, { size: BLANK_TARGET_ICON_SIZE }) }) : null,
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "2.1.4",
3
+ "version": "2.1.5",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {