@ultraviolet/ui 2.0.1 → 2.0.3

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 (89) hide show
  1. package/dist/components/Alert/index.cjs +10 -8
  2. package/dist/components/Alert/index.js +10 -8
  3. package/dist/components/Badge/index.cjs +7 -3
  4. package/dist/components/Badge/index.js +7 -3
  5. package/dist/components/Banner/index.cjs +10 -8
  6. package/dist/components/Banner/index.js +10 -8
  7. package/dist/components/Button/index.cjs +6 -4
  8. package/dist/components/Button/index.js +6 -4
  9. package/dist/components/Chip/index.cjs +10 -4
  10. package/dist/components/Chip/index.js +10 -4
  11. package/dist/components/DateInput/components/CalendarDaily.cjs +23 -9
  12. package/dist/components/DateInput/components/CalendarDaily.js +23 -9
  13. package/dist/components/DateInput/components/CalendarMonthly.cjs +7 -3
  14. package/dist/components/DateInput/components/CalendarMonthly.js +7 -3
  15. package/dist/components/DateInput/index.cjs +29 -11
  16. package/dist/components/DateInput/index.js +29 -11
  17. package/dist/components/Drawer/index.cjs +13 -9
  18. package/dist/components/Drawer/index.js +13 -9
  19. package/dist/components/Expandable/index.cjs +4 -2
  20. package/dist/components/Expandable/index.js +4 -2
  21. package/dist/components/GlobalAlert/index.cjs +5 -3
  22. package/dist/components/GlobalAlert/index.js +5 -3
  23. package/dist/components/InfiniteScroll/index.cjs +15 -5
  24. package/dist/components/InfiniteScroll/index.js +15 -5
  25. package/dist/components/LineChart/helpers.cjs +15 -5
  26. package/dist/components/LineChart/helpers.js +15 -5
  27. package/dist/components/Link/index.cjs +10 -6
  28. package/dist/components/Link/index.js +10 -6
  29. package/dist/components/List/ListContext.cjs +2 -4
  30. package/dist/components/List/ListContext.js +2 -4
  31. package/dist/components/Menu/MenuContent.cjs +14 -8
  32. package/dist/components/Menu/MenuContent.js +14 -8
  33. package/dist/components/Menu/MenuProvider.cjs +3 -1
  34. package/dist/components/Menu/MenuProvider.js +3 -1
  35. package/dist/components/Menu/components/Item.cjs +6 -4
  36. package/dist/components/Menu/components/Item.js +6 -4
  37. package/dist/components/Modal/ModalProvider.d.ts +2 -2
  38. package/dist/components/Modal/components/Dialog.cjs +6 -6
  39. package/dist/components/Modal/components/Dialog.js +6 -6
  40. package/dist/components/NumberInput/index.cjs +11 -7
  41. package/dist/components/NumberInput/index.js +11 -7
  42. package/dist/components/Pagination/index.cjs +3 -1
  43. package/dist/components/Pagination/index.js +3 -1
  44. package/dist/components/Popup/index.cjs +16 -12
  45. package/dist/components/Popup/index.js +16 -12
  46. package/dist/components/SelectInput/Dropdown.cjs +31 -20
  47. package/dist/components/SelectInput/Dropdown.d.ts +2 -1
  48. package/dist/components/SelectInput/Dropdown.js +32 -21
  49. package/dist/components/SelectInput/SearchBarDropdown.cjs +8 -4
  50. package/dist/components/SelectInput/SearchBarDropdown.js +8 -4
  51. package/dist/components/SelectInput/SelectBar.cjs +13 -10
  52. package/dist/components/SelectInput/SelectBar.d.ts +2 -1
  53. package/dist/components/SelectInput/SelectBar.js +13 -10
  54. package/dist/components/SelectInput/SelectInputProvider.cjs +14 -7
  55. package/dist/components/SelectInput/SelectInputProvider.js +14 -7
  56. package/dist/components/SelectInput/index.cjs +5 -4
  57. package/dist/components/SelectInput/index.js +5 -4
  58. package/dist/components/SelectableCard/index.cjs +18 -16
  59. package/dist/components/SelectableCard/index.js +18 -16
  60. package/dist/components/Slider/components/DoubleSlider.cjs +23 -11
  61. package/dist/components/Slider/components/DoubleSlider.js +23 -11
  62. package/dist/components/Slider/index.cjs +6 -2
  63. package/dist/components/Slider/index.js +6 -2
  64. package/dist/components/Stepper/index.cjs +12 -6
  65. package/dist/components/Stepper/index.js +12 -6
  66. package/dist/components/SwitchButton/index.cjs +13 -5
  67. package/dist/components/SwitchButton/index.js +13 -5
  68. package/dist/components/Tabs/Tab.cjs +8 -6
  69. package/dist/components/Tabs/Tab.js +8 -6
  70. package/dist/components/Tabs/TabMenu.cjs +5 -4
  71. package/dist/components/Tabs/TabMenu.js +5 -4
  72. package/dist/components/Tabs/index.cjs +6 -4
  73. package/dist/components/Tabs/index.js +6 -4
  74. package/dist/components/TimeInput/helpers.cjs +24 -8
  75. package/dist/components/TimeInput/helpers.js +24 -8
  76. package/dist/components/TimeInput/index.cjs +47 -21
  77. package/dist/components/TimeInput/index.js +47 -21
  78. package/dist/components/Toggle/index.cjs +23 -31
  79. package/dist/components/Toggle/index.js +24 -32
  80. package/dist/components/VerificationCode/index.cjs +13 -8
  81. package/dist/components/VerificationCode/index.js +13 -8
  82. package/dist/helpers/recursivelyGetChildrenString.cjs +9 -3
  83. package/dist/helpers/recursivelyGetChildrenString.js +9 -3
  84. package/dist/mocks/list.d.ts +2 -2
  85. package/dist/utils/responsive/utilities.cjs +6 -2
  86. package/dist/utils/responsive/utilities.js +6 -2
  87. package/dist/utils/searchAlgorithm.cjs +15 -5
  88. package/dist/utils/searchAlgorithm.js +15 -5
  89. package/package.json +7 -7
@@ -31,8 +31,12 @@ const InfiniteScroll = ({
31
31
  };
32
32
  }, []);
33
33
  const handleScroll = react.useCallback((scrollableContainer) => {
34
- if (!containerRef.current) return;
35
- if (!onLoadMore) return;
34
+ if (!containerRef.current) {
35
+ return;
36
+ }
37
+ if (!onLoadMore) {
38
+ return;
39
+ }
36
40
  const {
37
41
  scrollTop,
38
42
  scrollHeight,
@@ -52,7 +56,9 @@ const InfiniteScroll = ({
52
56
  }
53
57
  }, [onLoadMore, heightThreshold]);
54
58
  react.useEffect(() => {
55
- if (!hasMore) return;
59
+ if (!hasMore) {
60
+ return;
61
+ }
56
62
  let scrollableContainer = scrollParentRef?.current || containerRef.current?.parentElement;
57
63
  while (scrollableContainer && scrollableContainer !== document.body) {
58
64
  const {
@@ -63,7 +69,9 @@ const InfiniteScroll = ({
63
69
  }
64
70
  scrollableContainer = scrollableContainer.parentElement;
65
71
  }
66
- if (!scrollableContainer) return;
72
+ if (!scrollableContainer) {
73
+ return;
74
+ }
67
75
  const debouncedHandleScroll = debounce(() => handleScroll(scrollableContainer), 100);
68
76
  scrollableContainer.addEventListener("scroll", debouncedHandleScroll);
69
77
  return () => {
@@ -71,7 +79,9 @@ const InfiniteScroll = ({
71
79
  };
72
80
  }, [debounce, handleScroll, hasMore, onLoadMore, scrollParentRef]);
73
81
  const localLoader = react.useMemo(() => loader || /* @__PURE__ */ jsxRuntime.jsx(index.Loader, { active: true }), [loader]);
74
- if (isLoading) return localLoader;
82
+ if (isLoading) {
83
+ return localLoader;
84
+ }
75
85
  return /* @__PURE__ */ jsxRuntime.jsx(Component, { className, "data-testid": dataTestId, id, ref: containerRef, role: "feed", style: {
76
86
  height
77
87
  } });
@@ -29,8 +29,12 @@ const InfiniteScroll = ({
29
29
  };
30
30
  }, []);
31
31
  const handleScroll = useCallback((scrollableContainer) => {
32
- if (!containerRef.current) return;
33
- if (!onLoadMore) return;
32
+ if (!containerRef.current) {
33
+ return;
34
+ }
35
+ if (!onLoadMore) {
36
+ return;
37
+ }
34
38
  const {
35
39
  scrollTop,
36
40
  scrollHeight,
@@ -50,7 +54,9 @@ const InfiniteScroll = ({
50
54
  }
51
55
  }, [onLoadMore, heightThreshold]);
52
56
  useEffect(() => {
53
- if (!hasMore) return;
57
+ if (!hasMore) {
58
+ return;
59
+ }
54
60
  let scrollableContainer = scrollParentRef?.current || containerRef.current?.parentElement;
55
61
  while (scrollableContainer && scrollableContainer !== document.body) {
56
62
  const {
@@ -61,7 +67,9 @@ const InfiniteScroll = ({
61
67
  }
62
68
  scrollableContainer = scrollableContainer.parentElement;
63
69
  }
64
- if (!scrollableContainer) return;
70
+ if (!scrollableContainer) {
71
+ return;
72
+ }
65
73
  const debouncedHandleScroll = debounce(() => handleScroll(scrollableContainer), 100);
66
74
  scrollableContainer.addEventListener("scroll", debouncedHandleScroll);
67
75
  return () => {
@@ -69,7 +77,9 @@ const InfiniteScroll = ({
69
77
  };
70
78
  }, [debounce, handleScroll, hasMore, onLoadMore, scrollParentRef]);
71
79
  const localLoader = useMemo(() => loader || /* @__PURE__ */ jsx(Loader, { active: true }), [loader]);
72
- if (isLoading) return localLoader;
80
+ if (isLoading) {
81
+ return localLoader;
82
+ }
73
83
  return /* @__PURE__ */ jsx(Component, { className, "data-testid": dataTestId, id, ref: containerRef, role: "feed", style: {
74
84
  height
75
85
  } });
@@ -1,16 +1,24 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
3
  const parse = (data) => {
4
- if (typeof data === "number") return data || 0;
5
- if (typeof data === "string") return Number.parseFloat(data) || 0;
6
- if (data instanceof Date) return data.getTime();
4
+ if (typeof data === "number") {
5
+ return data || 0;
6
+ }
7
+ if (typeof data === "string") {
8
+ return Number.parseFloat(data) || 0;
9
+ }
10
+ if (data instanceof Date) {
11
+ return data.getTime();
12
+ }
7
13
  return 0;
8
14
  };
9
15
  const getMin = (values = []) => values.length > 0 ? Math.min(...values.map(parse)) : 0;
10
16
  const getMax = (values = []) => values.length > 0 ? Math.max(...values.map(parse)) : 0;
11
17
  const getAverage = (values = []) => values.length > 0 ? Math.round(values.reduce((sum, curr) => sum + parse(curr), 0) / values.length * 100) / 100 : 0;
12
18
  const getMaxChartValue = (preppedData) => {
13
- if (!preppedData?.length) return 0;
19
+ if (!preppedData?.length) {
20
+ return 0;
21
+ }
14
22
  const maximum = Math.max(...preppedData.map(({
15
23
  data
16
24
  }) => getMax(data.map(({
@@ -19,7 +27,9 @@ const getMaxChartValue = (preppedData) => {
19
27
  return Math.ceil(maximum + maximum * 0.1);
20
28
  };
21
29
  const getMinChartValue = (preppedData) => {
22
- if (!preppedData?.length) return 0;
30
+ if (!preppedData?.length) {
31
+ return 0;
32
+ }
23
33
  const minimum = Math.min(...preppedData.map(({
24
34
  data
25
35
  }) => getMin(data.map(({
@@ -1,14 +1,22 @@
1
1
  const parse = (data) => {
2
- if (typeof data === "number") return data || 0;
3
- if (typeof data === "string") return Number.parseFloat(data) || 0;
4
- if (data instanceof Date) return data.getTime();
2
+ if (typeof data === "number") {
3
+ return data || 0;
4
+ }
5
+ if (typeof data === "string") {
6
+ return Number.parseFloat(data) || 0;
7
+ }
8
+ if (data instanceof Date) {
9
+ return data.getTime();
10
+ }
5
11
  return 0;
6
12
  };
7
13
  const getMin = (values = []) => values.length > 0 ? Math.min(...values.map(parse)) : 0;
8
14
  const getMax = (values = []) => values.length > 0 ? Math.max(...values.map(parse)) : 0;
9
15
  const getAverage = (values = []) => values.length > 0 ? Math.round(values.reduce((sum, curr) => sum + parse(curr), 0) / values.length * 100) / 100 : 0;
10
16
  const getMaxChartValue = (preppedData) => {
11
- if (!preppedData?.length) return 0;
17
+ if (!preppedData?.length) {
18
+ return 0;
19
+ }
12
20
  const maximum = Math.max(...preppedData.map(({
13
21
  data
14
22
  }) => getMax(data.map(({
@@ -17,7 +25,9 @@ const getMaxChartValue = (preppedData) => {
17
25
  return Math.ceil(maximum + maximum * 0.1);
18
26
  };
19
27
  const getMinChartValue = (preppedData) => {
20
- if (!preppedData?.length) return 0;
28
+ if (!preppedData?.length) {
29
+ return 0;
30
+ }
21
31
  const minimum = Math.min(...preppedData.map(({
22
32
  data
23
33
  }) => getMin(data.map(({
@@ -17,7 +17,7 @@ const StyledArrowLeftIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowL
17
17
  label: "StyledArrowLeftIcon"
18
18
  })("margin-right:", ({
19
19
  theme
20
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') 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          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
20
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
21
21
  const StyledArrowRightIcon = /* @__PURE__ */ _styled__default.default(Icon.ArrowRightIcon, process.env.NODE_ENV === "production" ? {
22
22
  target: "e1afnb7a2"
23
23
  } : {
@@ -25,7 +25,7 @@ const StyledArrowRightIcon = /* @__PURE__ */ _styled__default.default(Icon.Arrow
25
25
  label: "StyledArrowRightIcon"
26
26
  })("margin-left:", ({
27
27
  theme
28
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAyBmD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') 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          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
28
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAyBmD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
29
29
  const StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(Icon.OpenInNewIcon, process.env.NODE_ENV === "production" ? {
30
30
  target: "e1afnb7a4"
31
31
  } : {
@@ -48,7 +48,7 @@ const StyledExternalIconContainer = /* @__PURE__ */ _styled__default.default("sp
48
48
  label: "StyledExternalIconContainer"
49
49
  })("display:inline-flex;padding-bottom:", ({
50
50
  theme
51
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAkE+C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') 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          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
51
+ }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAkE+C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
52
52
  const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NODE_ENV === "production" ? {
53
53
  shouldForwardProp: (prop) => !["sentiment", "iconPosition", "as", "oneLine"].includes(prop),
54
54
  target: "e1afnb7a0"
@@ -116,7 +116,7 @@ const StyledLink = /* @__PURE__ */ _styled__default.default("a", process.env.NOD
116
116
  }, ";}}&[data-variant='inline']{text-decoration:underline;text-decoration-thickness:1px;}&:hover::after,&:focus::after{background-color:", ({
117
117
  theme,
118
118
  sentiment
119
- }) => theme.colors[sentiment]?.text ?? theme.colors.neutral.text, ";}&:active{text-decoration-thickness:2px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAgFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') 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          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
119
+ }) => theme.colors[sentiment]?.text ?? theme.colors.neutral.text, ";}&:active{text-decoration-thickness:2px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx"],"names":[],"mappings":"AAgFE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Link/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport {\n  ArrowLeftIcon,\n  ArrowRightIcon,\n  OpenInNewIcon,\n} from '@ultraviolet/icons'\nimport type {\n  AnchorHTMLAttributes,\n  ForwardedRef,\n  HTMLAttributeAnchorTarget,\n  MouseEventHandler,\n  ReactNode,\n  RefObject,\n} from 'react'\nimport { forwardRef, useEffect, useMemo, useRef, useState } from 'react'\nimport recursivelyGetChildrenString from '../../helpers/recursivelyGetChildrenString'\nimport capitalize from '../../utils/capitalize'\nimport { Tooltip } from '../Tooltip'\n\nconst StyledArrowLeftIcon = styled(ArrowLeftIcon)`\n  margin-right: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledArrowRightIcon = styled(ArrowRightIcon)`\n  margin-left: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledOpenInNewIcon = StyledArrowRightIcon.withComponent(OpenInNewIcon)\n\nexport const PROMINENCES = {\n  default: '',\n  strong: 'strong',\n  stronger: 'stronger',\n  weak: 'weak',\n}\n\nexport type ProminenceProps = keyof typeof PROMINENCES\n\ntype LinkSizes = 'large' | 'small' | 'xsmall'\ntype LinkIconPosition = 'left' | 'right'\ntype LinkProps = {\n  children: ReactNode\n  target?: HTMLAttributeAnchorTarget\n  download?: string | boolean\n  sentiment?: 'primary' | 'info'\n  prominence?: ProminenceProps\n  size?: LinkSizes\n  iconPosition?: LinkIconPosition\n  rel?: AnchorHTMLAttributes<HTMLAnchorElement>['rel']\n  className?: string\n  href: string\n  // For react router shouldn't be used directly\n  onClick?: MouseEventHandler<HTMLAnchorElement>\n  'aria-label'?: string\n  'aria-current'?: AnchorHTMLAttributes<HTMLAnchorElement>['aria-current']\n  oneLine?: boolean\n  'data-testid'?: string\n  variant?: 'inline' | 'standalone'\n}\n\nconst ICON_SIZE = 'small'\nconst BLANK_TARGET_ICON_SIZE = 'small'\nconst TRANSITION_DURATION = 250\n\nconst StyledExternalIconContainer = styled.span`\n  display: inline-flex;\n  padding-bottom: ${({ theme }) => theme.space['0.5']};\n`\n\nconst StyledLink = styled('a', {\n  shouldForwardProp: prop =>\n    !['sentiment', 'iconPosition', 'as', 'oneLine'].includes(prop),\n})<{\n  sentiment: 'primary' | 'info'\n  prominence?: ProminenceProps\n  variant: 'captionStrong' | 'bodySmallStrong' | 'bodyStrong'\n  iconPosition?: LinkIconPosition\n  oneLine?: boolean\n}>`\n  background-color: transparent;\n  border: none;\n  padding: 0;\n  color: ${({ theme, sentiment, prominence }) => {\n    const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n    const themeColor = theme.colors[sentiment]\n    const text = `text${definedProminence}` as keyof typeof themeColor\n\n    return theme.colors[sentiment]?.[text] ?? theme.colors.neutral.text\n  }};\n  text-decoration: underline;\n  text-decoration-thickness: 1px;\n  text-underline-offset: 2px;\n  text-decoration-color: transparent;\n  transition: text-decoration-color ${TRANSITION_DURATION}ms ease-out;\n\n  ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n    transition: transform ${TRANSITION_DURATION}ms ease-out;\n  }\n\n  gap: ${({ theme }) => theme.space['1']};\n  position: relative;\n  cursor: pointer;\n\n  > * {\n    // Safari issue when something is inside an anchor\n    pointer-events: none;\n  }\n\n  ${({ oneLine }) =>\n    oneLine\n      ? `white-space: nowrap;\n    text-overflow: ellipsis;\n    overflow: hidden;\n    display: block;`\n      : 'width: fit-content;'}\n\n  ${({ variant, theme }) => `\n      font-size: ${theme.typography[variant].fontSize};\n      font-family: ${theme.typography[variant].fontFamily};\n      font-weight: ${theme.typography[variant].weight};\n      letter-spacing: ${theme.typography[variant].letterSpacing};\n      line-height: ${theme.typography[variant].lineHeight};\n      paragraph-spacing: ${theme.typography[variant].paragraphSpacing};\n      text-case: ${theme.typography[variant].textCase};\n    `}\n\n\n  &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n\n  &:hover,\n  &:focus {\n    ${StyledArrowLeftIcon}, ${StyledArrowRightIcon}, ${StyledOpenInNewIcon} {\n      transform: ${({ theme, iconPosition }) =>\n        iconPosition === 'left'\n          ? `translate(${theme.space['0.25']}, 0)`\n          : `translate(-${theme.space['0.25']}, 0)`};\n    }\n\n    outline: none;\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n    ${({ theme, sentiment, prominence }) => {\n      const definedProminence = capitalize(PROMINENCES[prominence ?? 'default'])\n\n      const themeColor = theme.colors[sentiment]\n\n      const text = `text${definedProminence}Hover` as keyof typeof themeColor\n\n      return `\n        color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };\n        text-decoration-color: ${\n          theme.colors[sentiment]?.[text] ?? theme.colors.neutral.textHover\n        };`\n    }}\n\n    &:visited {\n      text-decoration-color: transparent;\n\n      color: ${({ theme, prominence }) => {\n        const definedProminence = capitalize(\n          PROMINENCES[prominence ?? 'default'],\n        )\n        const themeColor = theme.colors.primary\n        const text = `text${definedProminence}` as keyof typeof themeColor\n\n        return theme.colors.primary[text] ?? theme.colors.primary.text\n      }};\n  }\n\n  }\n\n  &[data-variant='inline'] {\n    text-decoration: underline;\n    text-decoration-thickness: 1px;\n  }\n\n  &:hover::after,\n  &:focus::after {\n    background-color: ${({ theme, sentiment }) =>\n      theme.colors[sentiment]?.text ?? theme.colors.neutral.text};\n  }\n\n  &:active {\n    text-decoration-thickness: 2px;\n  }\n`\n\n/**\n * Link is a component used to navigate between pages or to external websites.\n */\nexport const Link = forwardRef(\n  (\n    {\n      children,\n      href,\n      target,\n      download,\n      sentiment = 'info',\n      prominence,\n      size = 'large',\n      iconPosition,\n      rel,\n      className,\n      onClick,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      oneLine = false,\n      'data-testid': dataTestId,\n      variant = 'standalone',\n    }: LinkProps,\n    ref: ForwardedRef<HTMLAnchorElement>,\n  ) => {\n    const isBlank = target === '_blank'\n    const computedRel = rel || (isBlank ? 'noopener noreferrer' : undefined)\n    const [isTruncated, setIsTruncated] = useState(false)\n    const elementRef = useRef<HTMLAnchorElement>(null)\n\n    const usedRef = (ref as RefObject<HTMLAnchorElement>) ?? elementRef\n\n    const finalStringChildren = recursivelyGetChildrenString(children)\n    const textVariant = useMemo(() => {\n      if (size === 'xsmall') {\n        return 'captionStrong'\n      }\n      if (size === 'small') {\n        return 'bodySmallStrong'\n      }\n\n      return 'bodyStrong'\n    }, [size])\n    useEffect(() => {\n      if (oneLine && usedRef?.current) {\n        const { offsetWidth, scrollWidth } = usedRef.current\n        setIsTruncated(offsetWidth < scrollWidth)\n      }\n    }, [oneLine, ref, usedRef])\n\n    return (\n      <Tooltip text={oneLine && isTruncated ? finalStringChildren : ''}>\n        <StyledLink\n          aria-current={ariaCurrent}\n          aria-label={ariaLabel}\n          className={className}\n          data-testid={dataTestId}\n          data-variant={variant}\n          download={download}\n          href={href}\n          iconPosition={iconPosition}\n          onClick={onClick}\n          oneLine={oneLine}\n          prominence={prominence}\n          ref={usedRef}\n          rel={computedRel}\n          sentiment={sentiment}\n          target={target}\n          variant={textVariant}\n        >\n          {!isBlank && iconPosition === 'left' ? (\n            <StyledArrowLeftIcon size={ICON_SIZE} />\n          ) : null}\n          {children}\n\n          {isBlank ? (\n            <StyledExternalIconContainer>\n              <StyledOpenInNewIcon size={BLANK_TARGET_ICON_SIZE} />\n            </StyledExternalIconContainer>\n          ) : null}\n\n          {!isBlank && iconPosition === 'right' ? (\n            <StyledArrowRightIcon size={ICON_SIZE} />\n          ) : null}\n        </StyledLink>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
120
120
  const Link = react.forwardRef(({
121
121
  children,
122
122
  href,
@@ -142,8 +142,12 @@ const Link = react.forwardRef(({
142
142
  const usedRef = ref ?? elementRef;
143
143
  const finalStringChildren = recursivelyGetChildrenString(children);
144
144
  const textVariant = react.useMemo(() => {
145
- if (size === "xsmall") return "captionStrong";
146
- if (size === "small") return "bodySmallStrong";
145
+ if (size === "xsmall") {
146
+ return "captionStrong";
147
+ }
148
+ if (size === "small") {
149
+ return "bodySmallStrong";
150
+ }
147
151
  return "bodyStrong";
148
152
  }, [size]);
149
153
  react.useEffect(() => {