@ultraviolet/ui 2.0.0-beta.4 → 2.0.0-beta.6

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.
Files changed (32) hide show
  1. package/dist/components/Banner/index.cjs +7 -7
  2. package/dist/components/Banner/index.js +7 -7
  3. package/dist/components/Button/index.cjs +18 -7
  4. package/dist/components/Button/index.js +20 -9
  5. package/dist/components/Checkbox/index.cjs +10 -10
  6. package/dist/components/Checkbox/index.js +10 -10
  7. package/dist/components/DateInput/index.cjs +2 -2
  8. package/dist/components/DateInput/index.d.ts +2 -2
  9. package/dist/components/DateInput/index.js +2 -2
  10. package/dist/components/ExpandableCard/index.cjs +134 -26
  11. package/dist/components/ExpandableCard/index.d.ts +25 -4
  12. package/dist/components/ExpandableCard/index.js +136 -28
  13. package/dist/components/InfiniteScroll/index.cjs +1 -1
  14. package/dist/components/InfiniteScroll/index.js +2 -2
  15. package/dist/components/Link/index.cjs +32 -20
  16. package/dist/components/Link/index.js +32 -20
  17. package/dist/components/Loader/index.cjs +25 -26
  18. package/dist/components/Loader/index.d.ts +16 -11
  19. package/dist/components/Loader/index.js +26 -27
  20. package/dist/components/Popover/index.cjs +4 -4
  21. package/dist/components/Popover/index.js +4 -4
  22. package/dist/components/Table/index.cjs +5 -4
  23. package/dist/components/Table/index.js +5 -4
  24. package/dist/components/Tag/index.cjs +15 -5
  25. package/dist/components/Tag/index.js +15 -5
  26. package/dist/components/TagList/index.cjs +6 -6
  27. package/dist/components/TagList/index.js +6 -6
  28. package/dist/components/TextInputV2/index.cjs +13 -9
  29. package/dist/components/TextInputV2/index.js +13 -9
  30. package/dist/components/TimeInputV2/index.cjs +6 -4
  31. package/dist/components/TimeInputV2/index.js +6 -4
  32. package/package.json +11 -11
@@ -11,20 +11,22 @@ const index = require("../Tooltip/index.cjs");
11
11
  const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
12
12
  const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
13
13
  const StyledArrowLeftIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowLeftIcon, process.env.NODE_ENV === "production" ? {
14
- target: "e1afnb7a2"
14
+ target: "e1afnb7a3"
15
15
  } : {
16
- target: "e1afnb7a2",
16
+ target: "e1afnb7a3",
17
17
  label: "StyledArrowLeftIcon"
18
- })("margin-left:", ({
18
+ })("margin-right:", ({
19
19
  theme
20
- }) => theme.space["1"], ";" + (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":"AAsBiD","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 type { ExtendedColor } from '../../theme'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-left: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(ArrowRightIcon)\nconst StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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      color: ${({ theme }) => theme.colors.primary.text};\n      text-decoration-color: transparent;\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.5']}, 0)`\n          : `translate(-${theme.space['0.5']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};\n      `\n    }}\n\n    &:visited {\n      color: ${({ theme }) => theme.colors.primary.textHover};\n      text-decoration-color: ${({ theme }) => theme.colors.primary.textHover};\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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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
- const StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(Icon.ArrowRightIcon, process.env.NODE_ENV === "production" ? {
22
- target: "e1afnb7a3"
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":"AAsBiD","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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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
+ const StyledArrowRightIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowRightIcon, process.env.NODE_ENV === "production" ? {
22
+ target: "e1afnb7a2"
23
23
  } : {
24
- target: "e1afnb7a3",
24
+ target: "e1afnb7a2",
25
25
  label: "StyledArrowRightIcon"
26
- });
27
- const StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(Icon.OpenInNewIcon, process.env.NODE_ENV === "production" ? {
26
+ })("margin-left:", ({
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":"AA0BmD","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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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
+ const StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(Icon.OpenInNewIcon, process.env.NODE_ENV === "production" ? {
28
30
  target: "e1afnb7a4"
29
31
  } : {
30
32
  target: "e1afnb7a4",
@@ -46,7 +48,7 @@ const StyledExternalIconContainer = /* @__PURE__ */ _styled__default.default("sp
46
48
  label: "StyledExternalIconContainer"
47
49
  })("display:inline-flex;padding-bottom:", ({
48
50
  theme
49
- }) => 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":"AAyE+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 type { ExtendedColor } from '../../theme'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-left: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(ArrowRightIcon)\nconst StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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      color: ${({ theme }) => theme.colors.primary.text};\n      text-decoration-color: transparent;\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.5']}, 0)`\n          : `translate(-${theme.space['0.5']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};\n      `\n    }}\n\n    &:visited {\n      color: ${({ theme }) => theme.colors.primary.textHover};\n      text-decoration-color: ${({ theme }) => theme.colors.primary.textHover};\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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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":"AA4E+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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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"]} */"));
50
52
  const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NODE_ENV === "production" ? {
51
53
  shouldForwardProp: (prop) => !["sentiment", "iconPosition", "as", "oneLine"].includes(prop),
52
54
  target: "e1afnb7a0"
@@ -85,12 +87,18 @@ const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NOD
85
87
  line-height: ${theme.typography[variant].lineHeight};
86
88
  paragraph-spacing: ${theme.typography[variant].paragraphSpacing};
87
89
  text-case: ${theme.typography[variant].textCase};
88
- `, " &:visited{color:", ({
89
- theme
90
- }) => theme.colors.primary.text, ";text-decoration-color:transparent;}&:hover,&:focus{outline:none;text-decoration:underline;text-decoration-thickness:1px;", StyledArrowLeftIcon, ",", StyledArrowRightIcon, ",", StyledOpenInNewIcon, "{transform:", ({
90
+ `, " &:visited{text-decoration-color:transparent;color:", ({
91
+ theme,
92
+ prominence
93
+ }) => {
94
+ const definedProminence = capitalize(PROMINENCES[prominence ?? "default"]);
95
+ theme.colors.primary;
96
+ const text = `text${definedProminence}`;
97
+ return theme.colors.primary[text] ?? theme.colors.primary.text;
98
+ }, ";}&:hover,&:focus{outline:none;text-decoration:underline;text-decoration-thickness:1px;", StyledArrowLeftIcon, ",", StyledArrowRightIcon, ",", StyledOpenInNewIcon, "{transform:", ({
91
99
  theme,
92
100
  iconPosition
93
- }) => iconPosition === "left" ? `translate(${theme.space["0.5"]}, 0)` : `translate(-${theme.space["0.5"]}, 0)`, ";}", ({
101
+ }) => iconPosition === "left" ? `translate(${theme.space["0.25"]}, 0)` : `translate(-${theme.space["0.25"]}, 0)`, ";}", ({
94
102
  theme,
95
103
  sentiment,
96
104
  prominence
@@ -108,11 +116,15 @@ const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NOD
108
116
  color: ${theme.colors.other.monochrome[sentiment].textHover};
109
117
  text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};
110
118
  `;
111
- }, " &:visited{color:", ({
112
- theme
113
- }) => theme.colors.primary.textHover, ";text-decoration-color:", ({
114
- theme
115
- }) => theme.colors.primary.textHover, ";}}&[data-variant='inline']{text-decoration:underline;text-decoration-thickness:1px;}&:hover::after,&:focus::after{background-color:", ({
119
+ }, " &:visited{text-decoration-color:transparent;color:", ({
120
+ theme,
121
+ prominence
122
+ }) => {
123
+ const definedProminence = capitalize(PROMINENCES[prominence ?? "default"]);
124
+ theme.colors.primary;
125
+ const text = `text${definedProminence}`;
126
+ return theme.colors.primary[text] ?? theme.colors.primary.text;
127
+ }, ";}}&[data-variant='inline']{text-decoration:underline;text-decoration-thickness:1px;}&:hover::after,&:focus::after{background-color:", ({
116
128
  theme,
117
129
  sentiment
118
130
  }) => {
@@ -121,7 +133,7 @@ const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NOD
121
133
  return theme.colors[sentiment]?.text ?? theme.colors.neutral.text;
122
134
  }
123
135
  return theme.colors.other.monochrome[sentiment].text;
124
- }, ";}&: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":"AAuFE","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 type { ExtendedColor } from '../../theme'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-left: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(ArrowRightIcon)\nconst StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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      color: ${({ theme }) => theme.colors.primary.text};\n      text-decoration-color: transparent;\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.5']}, 0)`\n          : `translate(-${theme.space['0.5']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};\n      `\n    }}\n\n    &:visited {\n      color: ${({ theme }) => theme.colors.primary.textHover};\n      text-decoration-color: ${({ theme }) => theme.colors.primary.textHover};\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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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"]} */"));
136
+ }, ";}&: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":"AA0FE","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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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"]} */"));
125
137
  const Link = React.forwardRef(({
126
138
  children,
127
139
  href,
@@ -7,20 +7,22 @@ import recursivelyGetChildrenString from "../../helpers/recursivelyGetChildrenSt
7
7
  import capitalize from "../../utils/capitalize.js";
8
8
  import { Tooltip } from "../Tooltip/index.js";
9
9
  const StyledArrowLeftIcon = /* @__PURE__ */ _styled(ArrowLeftIcon, process.env.NODE_ENV === "production" ? {
10
- target: "e1afnb7a2"
10
+ target: "e1afnb7a3"
11
11
  } : {
12
- target: "e1afnb7a2",
12
+ target: "e1afnb7a3",
13
13
  label: "StyledArrowLeftIcon"
14
- })("margin-left:", ({
14
+ })("margin-right:", ({
15
15
  theme
16
- }) => theme.space["1"], ";" + (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":"AAsBiD","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 type { ExtendedColor } from '../../theme'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-left: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(ArrowRightIcon)\nconst StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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      color: ${({ theme }) => theme.colors.primary.text};\n      text-decoration-color: transparent;\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.5']}, 0)`\n          : `translate(-${theme.space['0.5']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};\n      `\n    }}\n\n    &:visited {\n      color: ${({ theme }) => theme.colors.primary.textHover};\n      text-decoration-color: ${({ theme }) => theme.colors.primary.textHover};\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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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
- const StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(ArrowRightIcon, process.env.NODE_ENV === "production" ? {
18
- target: "e1afnb7a3"
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":"AAsBiD","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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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
+ const StyledArrowRightIcon = /* @__PURE__ */ _styled(ArrowRightIcon, process.env.NODE_ENV === "production" ? {
18
+ target: "e1afnb7a2"
19
19
  } : {
20
- target: "e1afnb7a3",
20
+ target: "e1afnb7a2",
21
21
  label: "StyledArrowRightIcon"
22
- });
23
- const StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(OpenInNewIcon, process.env.NODE_ENV === "production" ? {
22
+ })("margin-left:", ({
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":"AA0BmD","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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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
+ const StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon, process.env.NODE_ENV === "production" ? {
24
26
  target: "e1afnb7a4"
25
27
  } : {
26
28
  target: "e1afnb7a4",
@@ -42,7 +44,7 @@ const StyledExternalIconContainer = /* @__PURE__ */ _styled("span", process.env.
42
44
  label: "StyledExternalIconContainer"
43
45
  })("display:inline-flex;padding-bottom:", ({
44
46
  theme
45
- }) => 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":"AAyE+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 type { ExtendedColor } from '../../theme'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-left: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(ArrowRightIcon)\nconst StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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      color: ${({ theme }) => theme.colors.primary.text};\n      text-decoration-color: transparent;\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.5']}, 0)`\n          : `translate(-${theme.space['0.5']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};\n      `\n    }}\n\n    &:visited {\n      color: ${({ theme }) => theme.colors.primary.textHover};\n      text-decoration-color: ${({ theme }) => theme.colors.primary.textHover};\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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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":"AA4E+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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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"]} */"));
46
48
  const StyledLink = /* @__PURE__ */ _styled("a", process.env.NODE_ENV === "production" ? {
47
49
  shouldForwardProp: (prop) => !["sentiment", "iconPosition", "as", "oneLine"].includes(prop),
48
50
  target: "e1afnb7a0"
@@ -81,12 +83,18 @@ const StyledLink = /* @__PURE__ */ _styled("a", process.env.NODE_ENV === "produc
81
83
  line-height: ${theme.typography[variant].lineHeight};
82
84
  paragraph-spacing: ${theme.typography[variant].paragraphSpacing};
83
85
  text-case: ${theme.typography[variant].textCase};
84
- `, " &:visited{color:", ({
85
- theme
86
- }) => theme.colors.primary.text, ";text-decoration-color:transparent;}&:hover,&:focus{outline:none;text-decoration:underline;text-decoration-thickness:1px;", StyledArrowLeftIcon, ",", StyledArrowRightIcon, ",", StyledOpenInNewIcon, "{transform:", ({
86
+ `, " &:visited{text-decoration-color:transparent;color:", ({
87
+ theme,
88
+ prominence
89
+ }) => {
90
+ const definedProminence = capitalize(PROMINENCES[prominence ?? "default"]);
91
+ theme.colors.primary;
92
+ const text = `text${definedProminence}`;
93
+ return theme.colors.primary[text] ?? theme.colors.primary.text;
94
+ }, ";}&:hover,&:focus{outline:none;text-decoration:underline;text-decoration-thickness:1px;", StyledArrowLeftIcon, ",", StyledArrowRightIcon, ",", StyledOpenInNewIcon, "{transform:", ({
87
95
  theme,
88
96
  iconPosition
89
- }) => iconPosition === "left" ? `translate(${theme.space["0.5"]}, 0)` : `translate(-${theme.space["0.5"]}, 0)`, ";}", ({
97
+ }) => iconPosition === "left" ? `translate(${theme.space["0.25"]}, 0)` : `translate(-${theme.space["0.25"]}, 0)`, ";}", ({
90
98
  theme,
91
99
  sentiment,
92
100
  prominence
@@ -104,11 +112,15 @@ const StyledLink = /* @__PURE__ */ _styled("a", process.env.NODE_ENV === "produc
104
112
  color: ${theme.colors.other.monochrome[sentiment].textHover};
105
113
  text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};
106
114
  `;
107
- }, " &:visited{color:", ({
108
- theme
109
- }) => theme.colors.primary.textHover, ";text-decoration-color:", ({
110
- theme
111
- }) => theme.colors.primary.textHover, ";}}&[data-variant='inline']{text-decoration:underline;text-decoration-thickness:1px;}&:hover::after,&:focus::after{background-color:", ({
115
+ }, " &:visited{text-decoration-color:transparent;color:", ({
116
+ theme,
117
+ prominence
118
+ }) => {
119
+ const definedProminence = capitalize(PROMINENCES[prominence ?? "default"]);
120
+ theme.colors.primary;
121
+ const text = `text${definedProminence}`;
122
+ return theme.colors.primary[text] ?? theme.colors.primary.text;
123
+ }, ";}}&[data-variant='inline']{text-decoration:underline;text-decoration-thickness:1px;}&:hover::after,&:focus::after{background-color:", ({
112
124
  theme,
113
125
  sentiment
114
126
  }) => {
@@ -117,7 +129,7 @@ const StyledLink = /* @__PURE__ */ _styled("a", process.env.NODE_ENV === "produc
117
129
  return theme.colors[sentiment]?.text ?? theme.colors.neutral.text;
118
130
  }
119
131
  return theme.colors.other.monochrome[sentiment].text;
120
- }, ";}&: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":"AAuFE","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 type { ExtendedColor } from '../../theme'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-left: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledArrowRightIcon = StyledArrowLeftIcon.withComponent(ArrowRightIcon)\nconst StyledOpenInNewIcon = StyledArrowLeftIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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      color: ${({ theme }) => theme.colors.primary.text};\n      text-decoration-color: transparent;\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.5']}, 0)`\n          : `translate(-${theme.space['0.5']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].textHover};\n      `\n    }}\n\n    &:visited {\n      color: ${({ theme }) => theme.colors.primary.textHover};\n      text-decoration-color: ${({ theme }) => theme.colors.primary.textHover};\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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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"]} */"));
132
+ }, ";}&: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":"AA0FE","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 type { ExtendedColor } from '../../theme'\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  weak: 'weak',\n  strong: 'strong',\n  stronger: 'stronger',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\ntype SupportedSentiments = 'primary'\n\n/**\n * @deprecated Only `primary` is supported\n */\ntype DeprecatedSentiments = Exclude<ExtendedColor, SupportedSentiments>\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  /**\n   * **Only sentiment `primary` is supported.**\n   * All the other sentiments are deprecated\n   */\n  sentiment?: SupportedSentiments | DeprecatedSentiments\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  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: ExtendedColor\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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n    if (!isMonochrome) {\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\n    return theme.colors.other.monochrome[sentiment].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 isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\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      return `\n        color: ${theme.colors.other.monochrome[sentiment].textHover};\n        text-decoration-color: ${theme.colors.other.monochrome[sentiment].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      const isMonochrome = sentiment === 'white' || sentiment === 'black'\n\n      if (!isMonochrome) {\n        return theme.colors[sentiment]?.text ?? theme.colors.neutral.text\n      }\n\n      return theme.colors.other.monochrome[sentiment].text\n    }};\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      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') return 'captionStrong'\n      if (size === 'small') return 'bodySmallStrong'\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          href={href}\n          target={target}\n          download={download}\n          ref={usedRef}\n          sentiment={sentiment}\n          prominence={prominence}\n          rel={computedRel}\n          className={className}\n          variant={textVariant}\n          onClick={onClick}\n          iconPosition={iconPosition}\n          aria-label={ariaLabel}\n          oneLine={oneLine}\n          data-testid={dataTestId}\n          data-variant={variant}\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"]} */"));
121
133
  const Link = forwardRef(({
122
134
  children,
123
135
  href,