@ultraviolet/ui 2.0.4 → 3.0.0-beta.0

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 (35) hide show
  1. package/README.md +10 -6
  2. package/dist/components/Button/constants.cjs +0 -17
  3. package/dist/components/Button/constants.js +1 -18
  4. package/dist/components/Button/index.cjs +18 -187
  5. package/dist/components/Button/index.d.ts +18 -60
  6. package/dist/components/Button/index.js +18 -185
  7. package/dist/components/Button/styles.css.cjs +8 -0
  8. package/dist/components/Button/styles.css.d.ts +82 -0
  9. package/dist/components/Button/styles.css.js +8 -0
  10. package/dist/components/Chip/index.d.ts +1 -1
  11. package/dist/components/Dialog/constants.d.ts +1 -1
  12. package/dist/components/Dialog/index.d.ts +8 -6
  13. package/dist/components/List/index.d.ts +1 -1
  14. package/dist/components/Menu/index.d.ts +1 -1
  15. package/dist/components/SelectInput/Dropdown.cjs +10 -10
  16. package/dist/components/SelectInput/Dropdown.js +10 -10
  17. package/dist/components/SelectInput/SelectBar.cjs +18 -12
  18. package/dist/components/SelectInput/SelectBar.js +19 -13
  19. package/dist/components/Slider/styles.d.ts +1 -1
  20. package/dist/components/Tabs/TabMenuItem.d.ts +1 -1
  21. package/dist/components/Tabs/index.d.ts +2 -2
  22. package/dist/components/Tooltip/index.d.ts +1 -1
  23. package/dist/index.cjs +3 -0
  24. package/dist/index.d.ts +1 -1
  25. package/dist/index.js +3 -0
  26. package/dist/theme/ThemeProvider.cjs +21 -0
  27. package/dist/theme/ThemeProvider.d.ts +737 -0
  28. package/dist/theme/ThemeProvider.js +21 -0
  29. package/dist/theme/index.d.ts +3 -2
  30. package/dist/themes/dist/vanilla/themes.css.js.vanilla.css.cjs +1 -0
  31. package/dist/themes/dist/vanilla/themes.css.js.vanilla.css.js +1 -0
  32. package/dist/themes/dist/vanilla/themes.css.ts.vanilla.css.js.vanilla.css.cjs +1 -0
  33. package/dist/themes/dist/vanilla/themes.css.ts.vanilla.css.js.vanilla.css.js +1 -0
  34. package/dist/ui.css +1 -1
  35. package/package.json +12 -4
package/README.md CHANGED
@@ -31,16 +31,20 @@ You will also need to import fonts in your project by adding:
31
31
 
32
32
  ```tsx
33
33
  import { Global, ThemeProvider, css } from '@emotion/react'
34
+ import { ThemeProvider as ThemeProviderUV } from '@ultraviolet/ui'
34
35
  import { Button, normalize, theme } from '@ultraviolet/ui'
36
+ import '@ultraviolet/ui/styles'
35
37
 
36
38
  const App = () => (
37
39
  <ThemeProvider theme={theme}>
38
- <Global
39
- styles={css`
40
- ${normalize()}
41
- `}
42
- />
43
- <Button onClick={() => console.log('clicked')}>Click Me</Button>
40
+ <ThemeProviderUV>
41
+ <Global
42
+ styles={css`
43
+ ${normalize()}
44
+ `}
45
+ />
46
+ <Button onClick={() => console.log('clicked')}>Click Me</Button>
47
+ </ThemeProviderUV>
44
48
  </ThemeProvider>
45
49
  )
46
50
  ```
@@ -2,23 +2,6 @@
2
2
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
3
  const SIZE_HEIGHT = {
4
4
  large: "600",
5
- // sizing key value from tokens
6
- medium: "500",
7
- small: "400",
8
5
  xsmall: "300"
9
6
  };
10
- const SIZE_PADDING_KEY = {
11
- large: 2,
12
- medium: 1.5,
13
- small: 1,
14
- xsmall: 0.5
15
- };
16
- const SIZE_GAP_KEY = {
17
- large: 1,
18
- medium: 1,
19
- small: 1,
20
- xsmall: 0.5
21
- };
22
- exports.SIZE_GAP_KEY = SIZE_GAP_KEY;
23
7
  exports.SIZE_HEIGHT = SIZE_HEIGHT;
24
- exports.SIZE_PADDING_KEY = SIZE_PADDING_KEY;
@@ -1,24 +1,7 @@
1
1
  const SIZE_HEIGHT = {
2
2
  large: "600",
3
- // sizing key value from tokens
4
- medium: "500",
5
- small: "400",
6
3
  xsmall: "300"
7
4
  };
8
- const SIZE_PADDING_KEY = {
9
- large: 2,
10
- medium: 1.5,
11
- small: 1,
12
- xsmall: 0.5
13
- };
14
- const SIZE_GAP_KEY = {
15
- large: 1,
16
- medium: 1,
17
- small: 1,
18
- xsmall: 0.5
19
- };
20
5
  export {
21
- SIZE_GAP_KEY,
22
- SIZE_HEIGHT,
23
- SIZE_PADDING_KEY
6
+ SIZE_HEIGHT
24
7
  };
@@ -2,191 +2,11 @@
2
2
  "use strict";
3
3
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
4
4
  const jsxRuntime = require("@emotion/react/jsx-runtime");
5
- const _styled = require("@emotion/styled/base");
6
- const react$1 = require("@emotion/react");
7
5
  const react = require("react");
6
+ const ThemeProvider = require("../../theme/ThemeProvider.cjs");
8
7
  const index$1 = require("../Loader/index.cjs");
9
8
  const index = require("../Tooltip/index.cjs");
10
- const constants = require("./constants.cjs");
11
- const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
12
- const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
13
- const FOCUS_RING_KEY = {
14
- black: "focusNeutral",
15
- danger: "focusDanger",
16
- info: "focusInfo",
17
- neutral: "focusNeutral",
18
- primary: "focusPrimary",
19
- // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine
20
- secondary: "focusPrimary",
21
- success: "focusSuccess",
22
- warning: "focusWarning",
23
- white: "focusNeutral"
24
- };
25
- const isMonochrome = (sentiment) => sentiment === "white" || sentiment === "black";
26
- const coreStyle = ({
27
- theme,
28
- size,
29
- sentiment,
30
- fullWidth,
31
- disabled
32
- }) => {
33
- const font = size === "large" ? theme.typography.bodyStrong : theme.typography.bodySmallStrong;
34
- let width = "auto";
35
- if (fullWidth) {
36
- width = "100%";
37
- }
38
- return `
39
- display: inline-flex;
40
- position: relative;
41
- height: ${theme.sizing[constants.SIZE_HEIGHT[size]]};
42
- padding: 0 ${theme.space[constants.SIZE_PADDING_KEY[size]]};
43
- flex-direction: row;
44
- gap: ${theme.space[constants.SIZE_GAP_KEY[size]]};
45
- border-radius: ${theme.radii.default};
46
- box-sizing: border-box;
47
- width: ${width};
48
- align-items: center;
49
- cursor: ${disabled ? "not-allowed" : "pointer"};
50
- justify-content: center;
51
- outline-offset: 2px;
52
- white-space: nowrap;
53
- text-decoration: none;
54
- &:hover {
55
- text-decoration: none;
56
- }
57
-
58
-
59
- ${disabled ? "" : `
60
- &:active {
61
- box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};
62
- }
63
- `}
64
-
65
- /* We can't use Text component because of button hover effect, so we need to duplicate */
66
- font-size: ${font.fontSize};
67
- font-family: ${font.fontFamily};
68
- font-weight: ${font.weight};
69
- letter-spacing: ${font.letterSpacing};
70
- line-height: ${font.lineHeight};
71
- paragraph-spacing: ${font.paragraphSpacing};
72
- text-case: ${font.textCase};
73
- `;
74
- };
75
- const StyledFilledButton = /* @__PURE__ */ _styled__default.default("button", process.env.NODE_ENV === "production" ? {
76
- shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
77
- target: "e112qvla2"
78
- } : {
79
- shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
80
- target: "e112qvla2",
81
- label: "StyledFilledButton"
82
- })((args) => coreStyle(args), " ", index$1.StyledCircle, "{stroke:transparent;}background:", ({
83
- theme,
84
- sentiment
85
- }) => !isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrong : theme.colors.other.monochrome[sentiment].background, ";border:none;color:", ({
86
- theme,
87
- sentiment
88
- }) => !isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].text, ";", ({
89
- theme,
90
- sentiment,
91
- disabled
92
- }) => disabled ? `
93
- background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};
94
- color:
95
- ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};
96
- ` : `
97
- &:hover, &:active
98
- {
99
- background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};
100
- color:
101
- ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].textHover};
102
- }
103
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAyGqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
104
- const StyledOutlinedButton = /* @__PURE__ */ _styled__default.default("button", process.env.NODE_ENV === "production" ? {
105
- shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
106
- target: "e112qvla1"
107
- } : {
108
- shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
109
- target: "e112qvla1",
110
- label: "StyledOutlinedButton"
111
- })((args) => coreStyle(args), " ", index$1.StyledCircle, "{stroke:transparent;}background:none;border:1px solid ", ({
112
- theme,
113
- sentiment
114
- }) => !isMonochrome(sentiment) ? theme.colors[sentiment][sentiment === "neutral" ? "borderStrong" : "border"] : theme.colors.other.monochrome[sentiment].border, ";color:", ({
115
- theme,
116
- sentiment
117
- }) => !isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text, ";", ({
118
- theme,
119
- sentiment,
120
- disabled
121
- }) => disabled ? `
122
- color:
123
- ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};
124
- border: 1px solid ${!isMonochrome(sentiment) ? theme.colors[sentiment][sentiment === "neutral" ? "borderStrongDisabled" : "borderDisabled"] : theme.colors.other.monochrome[sentiment].borderDisabled};
125
-
126
- ` : `
127
- &:hover, &:active
128
- {
129
- background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};
130
- color:
131
- ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].textHover};
132
- border: 1px solid ${!isMonochrome(sentiment) ? theme.colors[sentiment][sentiment === "neutral" ? "borderStrongHover" : "borderHover"] : theme.colors.other.monochrome[sentiment].borderHover};
133
-
134
- }
135
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AA0IqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
136
- const StyledGhostButton = /* @__PURE__ */ _styled__default.default("button", process.env.NODE_ENV === "production" ? {
137
- shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
138
- target: "e112qvla0"
139
- } : {
140
- shouldForwardProp: (prop) => !["size", "sentiment", "fullWidth"].includes(prop),
141
- target: "e112qvla0",
142
- label: "StyledGhostButton"
143
- })((args) => coreStyle(args), " ", index$1.StyledCircle, "{stroke:transparent;}background:none;border:none;color:", ({
144
- theme,
145
- sentiment
146
- }) => !isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text, ";", ({
147
- theme,
148
- sentiment,
149
- disabled
150
- }) => disabled ? `
151
- color:
152
- ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};
153
- ` : `
154
- &:hover, &:active
155
- {
156
- background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};
157
- color:
158
- ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === "white" ? "black" : "white"].textHover};
159
- }
160
- `, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx"],"names":[],"mappings":"AAiMqB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Button/index.tsx","sourcesContent":["'use client'\n\nimport type { Theme } from '@emotion/react'\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport type {\n  AriaRole,\n  ButtonHTMLAttributes,\n  MouseEventHandler,\n  ReactNode,\n  Ref,\n} from 'react'\nimport { forwardRef, useMemo } from 'react'\nimport type { ExtendedColor } from '../../theme'\nimport { Loader, StyledCircle } from '../Loader'\nimport { Tooltip } from '../Tooltip'\nimport { SIZE_GAP_KEY, SIZE_HEIGHT, SIZE_PADDING_KEY } from './constants'\n\ntype ButtonSize = keyof typeof SIZE_HEIGHT\n\nexport const buttonSizes = Object.keys(SIZE_HEIGHT) as ButtonSize[]\n\n// FOCUS RING\nconst FOCUS_RING_KEY = {\n  black: 'focusNeutral',\n  danger: 'focusDanger',\n  info: 'focusInfo',\n  neutral: 'focusNeutral',\n  primary: 'focusPrimary',\n  // @note: no focusSecondary so far, it will be added later, so far focusPrimary sounds fine\n  secondary: 'focusPrimary',\n  success: 'focusSuccess',\n  warning: 'focusWarning',\n  white: 'focusNeutral',\n} as const\n\nconst isMonochrome = (sentiment: ExtendedColor) =>\n  sentiment === 'white' || sentiment === 'black'\n\n// VARIANTS\ntype StyledButtonProps = Required<\n  Pick<FinalProps, 'size' | 'sentiment' | 'disabled' | 'fullWidth'>\n>\n\nconst coreStyle = ({\n  theme,\n  size,\n  sentiment,\n  fullWidth,\n  disabled,\n}: { theme: Theme } & StyledButtonProps) => {\n  const font =\n    size === 'large'\n      ? theme.typography.bodyStrong\n      : theme.typography.bodySmallStrong\n\n  let width = 'auto'\n  if (fullWidth) {\n    width = '100%'\n  }\n\n  return `\n    display: inline-flex;\n    position: relative;\n    height: ${theme.sizing[SIZE_HEIGHT[size]]};\n    padding: 0 ${theme.space[SIZE_PADDING_KEY[size]]};\n    flex-direction: row;\n    gap: ${theme.space[SIZE_GAP_KEY[size]]};\n    border-radius: ${theme.radii.default};\n    box-sizing: border-box;\n    width: ${width};\n    align-items: center;\n    cursor: ${disabled ? 'not-allowed' : 'pointer'};\n    justify-content: center;\n    outline-offset: 2px;\n    white-space: nowrap;\n    text-decoration: none;\n    &:hover {\n      text-decoration: none;\n    }\n\n\n    ${\n      disabled\n        ? ''\n        : `\n            &:active {\n              box-shadow: ${theme.shadows[FOCUS_RING_KEY[sentiment]]};\n            }\n          `\n    }\n\n    /* We can't use Text component because of button hover effect, so we need to duplicate */\n    font-size: ${font.fontSize};\n    font-family: ${font.fontFamily};\n    font-weight: ${font.weight};\n    letter-spacing: ${font.letterSpacing};\n    line-height: ${font.lineHeight};\n    paragraph-spacing: ${font.paragraphSpacing};\n    text-case: ${font.textCase};\n  `\n}\n\nconst StyledFilledButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n    \n  background: ${({ theme, sentiment }) =>\n    !isMonochrome(sentiment)\n      ? theme.colors[sentiment].backgroundStrong\n      : theme.colors.other.monochrome[sentiment].background};\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].textStrong : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongDisabled : theme.colors.other.monochrome[sentiment].backgroundDisabled};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        `\n      : `\n            &:hover, &:active\n            {\n                background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundStrongHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n                color:\n                ${!isMonochrome(sentiment) ? theme.colors[sentiment].textStrongHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            }\n  `}\n`\n\nconst StyledOutlinedButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n  \n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: 1px solid\n    ${({ theme, sentiment }) =>\n      !isMonochrome(sentiment)\n        ? theme.colors[sentiment][\n            sentiment === 'neutral' ? 'borderStrong' : 'border'\n          ]\n        : theme.colors.other.monochrome[sentiment].border};\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n        border: 1px solid ${\n          !isMonochrome(sentiment)\n            ? theme.colors[sentiment][\n                sentiment === 'neutral'\n                  ? 'borderStrongDisabled'\n                  : 'borderDisabled'\n              ]\n            : theme.colors.other.monochrome[sentiment].borderDisabled\n        };\n\n    `\n      : `\n        &:hover, &:active\n       {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n            ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n            border: 1px solid ${\n              !isMonochrome(sentiment)\n                ? theme.colors[sentiment][\n                    sentiment === 'neutral'\n                      ? 'borderStrongHover'\n                      : 'borderHover'\n                  ]\n                : theme.colors.other.monochrome[sentiment].borderHover\n            };\n\n        }\n`};\n`\n\nconst StyledGhostButton = styled('button', {\n  shouldForwardProp: prop => !['size', 'sentiment', 'fullWidth'].includes(prop),\n})<StyledButtonProps>`\n  ${args => coreStyle(args)}\n\n  ${StyledCircle} {\n    stroke: transparent;\n  }\n\n  background: none;\n  border: none;\n  color: ${({ theme, sentiment }) => (!isMonochrome(sentiment) ? theme.colors[sentiment].text : theme.colors.other.monochrome[sentiment].text)};\n\n  ${({ theme, sentiment, disabled }) =>\n    disabled\n      ? `\n        color:\n          ${!isMonochrome(sentiment) ? theme.colors[sentiment].textDisabled : theme.colors.other.monochrome[sentiment].textDisabled};\n      `\n      : `\n        &:hover, &:active\n        {\n            background: ${!isMonochrome(sentiment) ? theme.colors[sentiment].backgroundHover : theme.colors.other.monochrome[sentiment].backgroundHover};\n            color:\n              ${!isMonochrome(sentiment) ? theme.colors[sentiment].textHover : theme.colors.other.monochrome[sentiment === 'white' ? 'black' : 'white'].textHover};\n        }\n`}\n`\n\nconst VARIANTS_COMPONENTS = {\n  filled: {\n    button: StyledFilledButton,\n    link: StyledFilledButton.withComponent('a'),\n  },\n  ghost: {\n    button: StyledGhostButton,\n    link: StyledGhostButton.withComponent('a'),\n  },\n  outlined: {\n    button: StyledOutlinedButton,\n    link: StyledOutlinedButton.withComponent('a'),\n  },\n}\n\ntype ButtonVariant = keyof typeof VARIANTS_COMPONENTS\nexport const buttonVariants = Object.keys(\n  VARIANTS_COMPONENTS,\n) as ButtonVariant[]\n\ntype CommonProps = {\n  type?: ButtonHTMLAttributes<HTMLButtonElement>['type']\n  autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus']\n  variant?: ButtonVariant\n  role?: AriaRole\n  size?: ButtonSize\n  className?: string\n  'data-testid'?: string\n  sentiment?: ExtendedColor\n  disabled?: boolean\n  fullWidth?: boolean\n  isLoading?: boolean\n  'aria-label'?: string\n  'aria-current'?: boolean\n  'aria-controls'?: string\n  'aria-expanded'?: boolean\n  'aria-haspopup'?: boolean\n  'aria-describedby'?: string\n  'aria-disabled'?: boolean\n  'aria-pressed'?: boolean\n  'aria-roledescription'?: string\n  onClick?: MouseEventHandler<HTMLElement>\n  tooltip?: string\n  tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex']\n  onMouseDown?: MouseEventHandler<HTMLElement>\n  onMouseUp?: MouseEventHandler<HTMLElement>\n  onMouseOut?: MouseEventHandler<HTMLElement>\n  onMouseEnter?: MouseEventHandler<HTMLElement>\n  onMouseLeave?: MouseEventHandler<HTMLElement>\n  onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown']\n  onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown']\n}\n\ntype FinalProps = CommonProps & {\n  children: ReactNode\n  name?: string\n  href?: string\n  target?: string\n  download?: string\n}\n\n/**\n * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,\n * canceling an action, or performing a delete operation.\n */\nexport const Button = forwardRef<Element, FinalProps>(\n  (\n    {\n      type = 'button',\n      className,\n      'data-testid': dataTestId,\n      sentiment = 'primary',\n      variant = 'filled',\n      size = 'large',\n      disabled = false,\n      fullWidth = false,\n      isLoading = false,\n      children,\n      onClick,\n      onMouseDown,\n      onMouseUp,\n      onMouseOut,\n      onMouseEnter,\n      onMouseLeave,\n      onPointerDown,\n      onKeyDown,\n      name,\n      'aria-label': ariaLabel,\n      'aria-current': ariaCurrent,\n      'aria-controls': ariaControls,\n      'aria-expanded': ariaExpanded,\n      'aria-haspopup': ariaHaspopup,\n      'aria-describedby': ariaDescribedby,\n      'aria-disabled': ariaDisabled,\n      'aria-pressed': ariaPressed,\n      'aria-roledescription': ariaRoledescription,\n      href,\n      download,\n      target,\n      role,\n      tooltip,\n      tabIndex,\n      autoFocus,\n    },\n    ref,\n  ) => {\n    const computeIsDisabled = disabled || isLoading\n    const { theme } = useTheme()\n    const computedSentimentLoader = useMemo(() => {\n      if (variant === 'filled' && !['black', 'white'].includes(sentiment)) {\n        if (theme === 'light') {\n          return 'white'\n        }\n\n        return 'black'\n      }\n\n      return sentiment\n    }, [sentiment, theme, variant])\n\n    const content = (\n      <>\n        {isLoading ? (\n          <Loader active sentiment={computedSentimentLoader} size=\"small\" />\n        ) : null}\n        {children}\n      </>\n    )\n\n    // @note: an anchor can't be disabled\n    if (href && !computeIsDisabled) {\n      const Component = VARIANTS_COMPONENTS[variant].link\n\n      return (\n        <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n          <Component\n            aria-controls={ariaControls}\n            aria-current={ariaCurrent}\n            aria-describedby={ariaDescribedby}\n            aria-disabled={ariaDisabled ?? disabled}\n            aria-expanded={ariaExpanded}\n            aria-haspopup={ariaHaspopup}\n            aria-label={ariaLabel}\n            aria-pressed={ariaPressed}\n            aria-roledescription={ariaRoledescription}\n            autoFocus={autoFocus}\n            className={className}\n            data-testid={dataTestId}\n            disabled={false}\n            download={download}\n            fullWidth={fullWidth}\n            href={href}\n            onClick={onClick}\n            onMouseDown={onMouseDown}\n            onMouseEnter={onMouseEnter}\n            onMouseLeave={onMouseLeave}\n            onMouseOut={onMouseOut}\n            onMouseUp={onMouseUp}\n            ref={ref as Ref<HTMLAnchorElement>}\n            role={role}\n            sentiment={sentiment}\n            size={size}\n            tabIndex={tabIndex}\n            target={target}\n            type={type}\n          >\n            {content}\n          </Component>\n        </Tooltip>\n      )\n    }\n\n    const Component = VARIANTS_COMPONENTS[variant].button\n\n    return (\n      <Tooltip containerFullWidth={fullWidth} text={tooltip}>\n        <Component\n          aria-controls={ariaControls}\n          aria-current={ariaCurrent}\n          aria-expanded={ariaExpanded}\n          aria-haspopup={ariaHaspopup}\n          aria-label={ariaLabel}\n          autoFocus={autoFocus}\n          className={className}\n          data-testid={dataTestId}\n          disabled={computeIsDisabled}\n          fullWidth={fullWidth}\n          name={name}\n          onClick={onClick}\n          onKeyDown={onKeyDown}\n          onMouseDown={onMouseDown}\n          onMouseEnter={onMouseEnter}\n          onMouseLeave={onMouseLeave}\n          onMouseOut={onMouseOut}\n          onMouseUp={onMouseUp}\n          onPointerDown={onPointerDown}\n          ref={ref as Ref<HTMLButtonElement>}\n          role={role}\n          sentiment={sentiment}\n          size={size}\n          tabIndex={tabIndex}\n          type={type}\n        >\n          {content}\n        </Component>\n      </Tooltip>\n    )\n  },\n)\n"]} */"));
161
- const VARIANTS_COMPONENTS = {
162
- filled: {
163
- button: StyledFilledButton,
164
- link: StyledFilledButton.withComponent("a", process.env.NODE_ENV === "production" ? {
165
- target: "e112qvla3"
166
- } : {
167
- target: "e112qvla3",
168
- label: "link"
169
- })
170
- },
171
- ghost: {
172
- button: StyledGhostButton,
173
- link: StyledGhostButton.withComponent("a", process.env.NODE_ENV === "production" ? {
174
- target: "e112qvla4"
175
- } : {
176
- target: "e112qvla4",
177
- label: "link"
178
- })
179
- },
180
- outlined: {
181
- button: StyledOutlinedButton,
182
- link: StyledOutlinedButton.withComponent("a", process.env.NODE_ENV === "production" ? {
183
- target: "e112qvla5"
184
- } : {
185
- target: "e112qvla5",
186
- label: "link"
187
- })
188
- }
189
- };
9
+ const styles_css = require("./styles.css.cjs");
190
10
  const Button = react.forwardRef(({
191
11
  type = "button",
192
12
  className,
@@ -202,6 +22,7 @@ const Button = react.forwardRef(({
202
22
  onMouseDown,
203
23
  onMouseUp,
204
24
  onMouseOut,
25
+ onBlur,
205
26
  onMouseEnter,
206
27
  onMouseLeave,
207
28
  onPointerDown,
@@ -227,7 +48,7 @@ const Button = react.forwardRef(({
227
48
  const computeIsDisabled = disabled || isLoading;
228
49
  const {
229
50
  theme
230
- } = react$1.useTheme();
51
+ } = ThemeProvider.useTheme();
231
52
  const computedSentimentLoader = react.useMemo(() => {
232
53
  if (variant === "filled" && !["black", "white"].includes(sentiment)) {
233
54
  if (theme === "light") {
@@ -242,10 +63,20 @@ const Button = react.forwardRef(({
242
63
  children
243
64
  ] });
244
65
  if (href && !computeIsDisabled) {
245
- const Component2 = VARIANTS_COMPONENTS[variant].link;
246
- return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsx(Component2, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-describedby": ariaDescribedby, "aria-disabled": ariaDisabled ?? disabled, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, "aria-pressed": ariaPressed, "aria-roledescription": ariaRoledescription, autoFocus, className, "data-testid": dataTestId, disabled: false, download, fullWidth, href, onClick, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, ref, role, sentiment, size, tabIndex, target, type, children: content }) });
66
+ return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsx("a", { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-describedby": ariaDescribedby, "aria-disabled": ariaDisabled ?? disabled, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, "aria-pressed": ariaPressed, "aria-roledescription": ariaRoledescription, autoFocus, className: `${className ?? ""} ${styles_css.button({
67
+ disabled,
68
+ fullWidth,
69
+ sentiment,
70
+ size,
71
+ variant
72
+ })}`, "data-testid": dataTestId, download, href, onBlur, onClick, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, ref, role, tabIndex, target, type, children: content }) });
247
73
  }
248
- const Component = VARIANTS_COMPONENTS[variant].button;
249
- return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsx(Component, { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, autoFocus, className, "data-testid": dataTestId, disabled: computeIsDisabled, fullWidth, name, onClick, onKeyDown, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, onPointerDown, ref, role, sentiment, size, tabIndex, type, children: content }) });
74
+ return /* @__PURE__ */ jsxRuntime.jsx(index.Tooltip, { containerFullWidth: fullWidth, text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsx("button", { "aria-controls": ariaControls, "aria-current": ariaCurrent, "aria-expanded": ariaExpanded, "aria-haspopup": ariaHaspopup, "aria-label": ariaLabel, autoFocus, className: `${className ?? ""} ${styles_css.button({
75
+ disabled,
76
+ fullWidth,
77
+ sentiment,
78
+ size,
79
+ variant
80
+ })}`, "data-testid": dataTestId, disabled: computeIsDisabled, name, onBlur, onClick, onKeyDown, onMouseDown, onMouseEnter, onMouseLeave, onMouseOut, onMouseUp, onPointerDown, ref, role, tabIndex, type, children: content }) });
250
81
  });
251
82
  exports.Button = Button;
@@ -1,54 +1,17 @@
1
- import type { Theme } from '@emotion/react';
2
1
  import type { AriaRole, ButtonHTMLAttributes, MouseEventHandler, ReactNode } from 'react';
3
- import type { ExtendedColor } from '../../theme';
4
2
  import { SIZE_HEIGHT } from './constants';
5
3
  type ButtonSize = keyof typeof SIZE_HEIGHT;
6
4
  export declare const buttonSizes: ButtonSize[];
7
- declare const VARIANTS_COMPONENTS: {
8
- filled: {
9
- button: import("@emotion/styled").StyledComponent<{
10
- theme?: Theme;
11
- as?: React.ElementType;
12
- } & Required<Pick<FinalProps, "size" | "sentiment" | "disabled" | "fullWidth">>, import("react").DetailedHTMLProps<ButtonHTMLAttributes<HTMLButtonElement>, HTMLButtonElement>, {}>;
13
- link: import("@emotion/styled").StyledComponent<{
14
- theme?: Theme;
15
- as?: React.ElementType;
16
- } & Required<Pick<FinalProps, "size" | "sentiment" | "disabled" | "fullWidth">>, import("react").DetailedHTMLProps<import("react").AnchorHTMLAttributes<HTMLAnchorElement>, HTMLAnchorElement>, {}>;
17
- };
18
- ghost: {
19
- button: import("@emotion/styled").StyledComponent<{
20
- theme?: Theme;
21
- as?: React.ElementType;
22
- } & Required<Pick<FinalProps, "size" | "sentiment" | "disabled" | "fullWidth">>, import("react").DetailedHTMLProps<ButtonHTMLAttributes<HTMLButtonElement>, HTMLButtonElement>, {}>;
23
- link: import("@emotion/styled").StyledComponent<{
24
- theme?: Theme;
25
- as?: React.ElementType;
26
- } & Required<Pick<FinalProps, "size" | "sentiment" | "disabled" | "fullWidth">>, import("react").DetailedHTMLProps<import("react").AnchorHTMLAttributes<HTMLAnchorElement>, HTMLAnchorElement>, {}>;
27
- };
28
- outlined: {
29
- button: import("@emotion/styled").StyledComponent<{
30
- theme?: Theme;
31
- as?: React.ElementType;
32
- } & Required<Pick<FinalProps, "size" | "sentiment" | "disabled" | "fullWidth">>, import("react").DetailedHTMLProps<ButtonHTMLAttributes<HTMLButtonElement>, HTMLButtonElement>, {}>;
33
- link: import("@emotion/styled").StyledComponent<{
34
- theme?: Theme;
35
- as?: React.ElementType;
36
- } & Required<Pick<FinalProps, "size" | "sentiment" | "disabled" | "fullWidth">>, import("react").DetailedHTMLProps<import("react").AnchorHTMLAttributes<HTMLAnchorElement>, HTMLAnchorElement>, {}>;
37
- };
38
- };
39
- type ButtonVariant = keyof typeof VARIANTS_COMPONENTS;
40
- export declare const buttonVariants: ButtonVariant[];
41
- type CommonProps = {
42
- type?: ButtonHTMLAttributes<HTMLButtonElement>['type'];
43
- autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>['autoFocus'];
44
- variant?: ButtonVariant;
5
+ /**
6
+ * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,
7
+ * canceling an action, or performing a delete operation.
8
+ */
9
+ export declare const Button: import("react").ForwardRefExoticComponent<{
10
+ type?: ButtonHTMLAttributes<HTMLButtonElement>["type"];
11
+ autoFocus?: ButtonHTMLAttributes<HTMLButtonElement>["autoFocus"];
45
12
  role?: AriaRole;
46
- size?: ButtonSize;
47
13
  className?: string;
48
14
  'data-testid'?: string;
49
- sentiment?: ExtendedColor;
50
- disabled?: boolean;
51
- fullWidth?: boolean;
52
15
  isLoading?: boolean;
53
16
  'aria-label'?: string;
54
17
  'aria-current'?: boolean;
@@ -61,27 +24,22 @@ type CommonProps = {
61
24
  'aria-roledescription'?: string;
62
25
  onClick?: MouseEventHandler<HTMLElement>;
63
26
  tooltip?: string;
64
- tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>['tabIndex'];
27
+ tabIndex?: ButtonHTMLAttributes<HTMLButtonElement>["tabIndex"];
65
28
  onMouseDown?: MouseEventHandler<HTMLElement>;
66
29
  onMouseUp?: MouseEventHandler<HTMLElement>;
67
30
  onMouseOut?: MouseEventHandler<HTMLElement>;
31
+ onBlur?: ButtonHTMLAttributes<HTMLElement>["onBlur"];
68
32
  onMouseEnter?: MouseEventHandler<HTMLElement>;
69
33
  onMouseLeave?: MouseEventHandler<HTMLElement>;
70
- onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>['onPointerDown'];
71
- onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>['onKeyDown'];
72
- };
73
- type FinalProps = CommonProps & {
74
- children: ReactNode;
75
- name?: string;
76
- href?: string;
77
- target?: string;
78
- download?: string;
79
- };
80
- /**
81
- * Button component is used to trigger an action or event, such as submitting a form, opening a dialog,
82
- * canceling an action, or performing a delete operation.
83
- */
84
- export declare const Button: import("react").ForwardRefExoticComponent<CommonProps & {
34
+ onPointerDown?: ButtonHTMLAttributes<HTMLButtonElement>["onPointerDown"];
35
+ onKeyDown?: ButtonHTMLAttributes<HTMLButtonElement>["onKeyDown"];
36
+ } & {
37
+ size?: "small" | "large" | "medium" | "xsmall" | undefined;
38
+ fullWidth?: boolean | undefined;
39
+ sentiment?: "danger" | "info" | "neutral" | "success" | "primary" | "warning" | "black" | "white" | "secondary" | undefined;
40
+ variant?: "filled" | "outlined" | "ghost" | undefined;
41
+ disabled?: boolean | undefined;
42
+ } & {
85
43
  children: ReactNode;
86
44
  name?: string;
87
45
  href?: string;