@ultraviolet/ui 1.51.2 → 1.51.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 (127) hide show
  1. package/dist/components/ActionBar/index.js +16 -26
  2. package/dist/components/Alert/index.js +77 -88
  3. package/dist/components/Avatar/index.js +29 -26
  4. package/dist/components/Badge/index.js +36 -52
  5. package/dist/components/Banner/index.js +64 -139
  6. package/dist/components/BarChart/Tooltip.js +23 -18
  7. package/dist/components/BarChart/index.js +29 -42
  8. package/dist/components/BarStack/index.js +89 -163
  9. package/dist/components/Breadcrumbs/index.js +30 -33
  10. package/dist/components/Bullet/index.js +24 -28
  11. package/dist/components/Button/index.js +117 -159
  12. package/dist/components/Card/index.js +45 -52
  13. package/dist/components/Carousel/index.js +75 -99
  14. package/dist/components/Checkbox/index.js +246 -343
  15. package/dist/components/CheckboxGroup/index.js +60 -56
  16. package/dist/components/CopyButton/index.js +1 -15
  17. package/dist/components/DateInput/index.js +124 -243
  18. package/dist/components/Dialog/Context.js +1 -3
  19. package/dist/components/Dialog/index.js +22 -43
  20. package/dist/components/Dialog/subComponents/Button.js +1 -10
  21. package/dist/components/Dialog/subComponents/Stack.js +3 -1
  22. package/dist/components/Dialog/subComponents/Text.js +3 -1
  23. package/dist/components/EmptyState/index.js +58 -50
  24. package/dist/components/Expandable/index.js +19 -23
  25. package/dist/components/GlobalAlert/GlobalAlertLink.js +4 -20
  26. package/dist/components/GlobalAlert/index.js +34 -87
  27. package/dist/components/LineChart/CustomLegend.js +76 -72
  28. package/dist/components/LineChart/Tooltip.js +26 -37
  29. package/dist/components/LineChart/helpers.js +11 -9
  30. package/dist/components/LineChart/index.js +38 -55
  31. package/dist/components/Link/index.js +86 -125
  32. package/dist/components/List/Body.js +12 -7
  33. package/dist/components/List/Cell.js +36 -33
  34. package/dist/components/List/HeaderCell.js +54 -68
  35. package/dist/components/List/HeaderRow.js +16 -18
  36. package/dist/components/List/ListContext.js +68 -87
  37. package/dist/components/List/Row.js +136 -169
  38. package/dist/components/List/SelectBar.js +31 -14
  39. package/dist/components/List/SkeletonRows.js +42 -23
  40. package/dist/components/List/index.js +39 -59
  41. package/dist/components/Loader/index.js +44 -65
  42. package/dist/components/Menu/Item.js +52 -62
  43. package/dist/components/Menu/index.js +105 -171
  44. package/dist/components/MenuV2/Group.js +14 -16
  45. package/dist/components/MenuV2/Item.js +67 -74
  46. package/dist/components/MenuV2/index.js +76 -110
  47. package/dist/components/Meter/index.js +44 -57
  48. package/dist/components/Modal/Dialog.js +44 -100
  49. package/dist/components/Modal/index.js +23 -58
  50. package/dist/components/Notice/index.js +13 -29
  51. package/dist/components/Notification/index.js +9 -48
  52. package/dist/components/NumberInput/index.js +113 -200
  53. package/dist/components/NumberInputV2/index.js +200 -349
  54. package/dist/components/Pagination/getPageNumbers.js +3 -1
  55. package/dist/components/Pagination/index.js +45 -81
  56. package/dist/components/PasswordCheck/index.js +10 -15
  57. package/dist/components/PasswordStrengthMeter/index.js +66 -68
  58. package/dist/components/PieChart/Legends.js +121 -99
  59. package/dist/components/PieChart/Tooltip.js +35 -15
  60. package/dist/components/PieChart/index.js +68 -87
  61. package/dist/components/Popover/index.js +57 -106
  62. package/dist/components/Popup/helpers.js +24 -12
  63. package/dist/components/Popup/index.js +263 -351
  64. package/dist/components/ProgressBar/index.js +33 -51
  65. package/dist/components/Radio/index.js +119 -188
  66. package/dist/components/RadioGroup/index.js +46 -51
  67. package/dist/components/Row/index.js +11 -20
  68. package/dist/components/SelectInput/index.js +165 -212
  69. package/dist/components/SelectInputV2/Dropdown.js +233 -473
  70. package/dist/components/SelectInputV2/DropdownOption.js +44 -115
  71. package/dist/components/SelectInputV2/SearchBarDropdown.js +28 -44
  72. package/dist/components/SelectInputV2/SelectBar.js +137 -240
  73. package/dist/components/SelectInputV2/SelectInputProvider.js +45 -77
  74. package/dist/components/SelectInputV2/index.js +39 -98
  75. package/dist/components/SelectableCard/index.js +127 -183
  76. package/dist/components/SelectableCardGroup/index.js +50 -55
  77. package/dist/components/Separator/index.js +44 -64
  78. package/dist/components/Skeleton/Block.js +28 -20
  79. package/dist/components/Skeleton/Blocks.js +26 -16
  80. package/dist/components/Skeleton/BoxWithIcon.js +24 -17
  81. package/dist/components/Skeleton/Donut.js +46 -21
  82. package/dist/components/Skeleton/IconSkeleton.js +19 -9
  83. package/dist/components/Skeleton/Line.js +11 -8
  84. package/dist/components/Skeleton/List.js +48 -21
  85. package/dist/components/Skeleton/Slider.js +36 -22
  86. package/dist/components/Skeleton/Square.js +9 -7
  87. package/dist/components/Skeleton/index.js +38 -45
  88. package/dist/components/Snippet/index.js +139 -128
  89. package/dist/components/Stack/index.js +10 -17
  90. package/dist/components/Status/index.js +42 -27
  91. package/dist/components/StepList/index.js +34 -45
  92. package/dist/components/Stepper/index.js +106 -114
  93. package/dist/components/SwitchButton/FocusOverlay.js +18 -27
  94. package/dist/components/SwitchButton/index.js +39 -103
  95. package/dist/components/Table/Body.js +3 -1
  96. package/dist/components/Table/Cell.js +15 -6
  97. package/dist/components/Table/Header.js +12 -5
  98. package/dist/components/Table/HeaderCell.js +48 -91
  99. package/dist/components/Table/HeaderRow.js +11 -13
  100. package/dist/components/Table/Row.js +39 -44
  101. package/dist/components/Table/SelectBar.js +31 -14
  102. package/dist/components/Table/SkeletonRows.js +43 -11
  103. package/dist/components/Table/TableContext.js +29 -44
  104. package/dist/components/Table/index.js +37 -51
  105. package/dist/components/Tabs/Tab.js +108 -158
  106. package/dist/components/Tabs/TabMenu.js +52 -53
  107. package/dist/components/Tabs/TabMenuItem.js +19 -24
  108. package/dist/components/Tabs/TabsContext.js +1 -3
  109. package/dist/components/Tabs/index.js +44 -79
  110. package/dist/components/Tag/index.js +66 -75
  111. package/dist/components/TagInput/index.js +125 -188
  112. package/dist/components/TagList/index.js +51 -56
  113. package/dist/components/Text/index.js +10 -35
  114. package/dist/components/TextArea/index.js +140 -208
  115. package/dist/components/TextInput/index.js +326 -411
  116. package/dist/components/TextInputV2/index.js +170 -294
  117. package/dist/components/TimeInput/index.js +3 -174
  118. package/dist/components/Toaster/index.js +26 -74
  119. package/dist/components/Toggle/index.js +112 -172
  120. package/dist/components/ToggleGroup/index.js +44 -35
  121. package/dist/components/Tooltip/index.js +34 -41
  122. package/dist/components/VerificationCode/index.js +58 -63
  123. package/dist/helpers/legend.js +4 -4
  124. package/dist/theme/index.js +10 -13
  125. package/dist/utils/ids.js +3 -1
  126. package/dist/utils/responsive/Breakpoint.js +8 -6
  127. package/package.json +4 -4
@@ -1,6 +1,6 @@
1
1
  import { jsx, jsxs } from "@emotion/react/jsx-runtime";
2
+ import _styled from "@emotion/styled/base";
2
3
  import { css } from "@emotion/react";
3
- import styled from "@emotion/styled";
4
4
  import randomName from "@scaleway/random-name";
5
5
  import { Icon } from "@ultraviolet/icons";
6
6
  import { forwardRef, useRef, useState, useCallback, useImperativeHandle, useEffect, useMemo } from "react";
@@ -10,6 +10,9 @@ import { Notice } from "../Notice/index.js";
10
10
  import { Separator } from "../Separator/index.js";
11
11
  import { Stack } from "../Stack/index.js";
12
12
  import { Text } from "../Text/index.js";
13
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() {
14
+ return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
15
+ }
13
16
  const inputSizes = {
14
17
  medium: {
15
18
  default: `
@@ -35,158 +38,187 @@ const inputSizes = {
35
38
  `
36
39
  }
37
40
  };
38
- const StyledSeparator = styled(Separator)`
39
- margin: 1px 0px;
40
- height: calc(100% - 2px);
41
- background-color: ${({ theme: { colors } }) => colors.neutral.backgroundStrong};
42
- `;
43
- const StyledRightElement = styled("div", {
44
- shouldForwardProp: (prop) => !["edit", "touchable"].includes(prop)
45
- })`
46
- ${({ theme: { colors, space } }) => css`
47
- pointer-events: none;
48
- position: absolute;
49
- right: 0;
50
- bottom: 0;
51
- top: 0;
52
- padding: 0 ${space["1"]};
53
- display: flex;
54
- gap: ${space["1"]};
55
- align-items: center;
56
- transition:
57
- transform 150ms,
58
- color 150ms;
59
- color: ${colors.neutral.textWeak};
60
-
61
- &:hover,
62
- &:focus-within {
63
- color: ${colors.neutral.textWeakHover};
64
- }
65
- `}
66
-
67
- ${({ touchable }) => touchable && css`
68
- pointer-events: auto;
69
- > button {
70
- box-shadow: none !important;
71
- }
72
- `}
73
- `;
74
- const StyledLabel = styled("label", {
75
- shouldForwardProp: (prop) => !["edit", "error", "resizable", "fillAvailable"].includes(prop)
76
- })`
77
- display: block;
78
- position: absolute;
79
- left: 0;
80
- top: 0;
81
- padding-left: 8px;
82
- padding-right: 8px;
83
- pointer-events: none;
84
- color: ${({ theme: { colors } }) => colors.neutral.textWeak};
85
- white-space: nowrap;
86
- overflow: hidden;
87
- text-overflow: ellipsis;
88
- width: 100%;
89
- height: 48px;
90
- font-size: ${({ theme }) => theme.typography.bodySmall};
91
- transition: transform 150ms;
92
- transform: translate(0, 12px) scale(1);
93
-
94
- ${({ edit }) => edit && css`
95
- transform: translate(-9.6%, -3px) scale(0.8);
96
- `}
97
-
98
- ${({ disabled, theme: { colors } }) => disabled && css`
99
- color: ${colors.neutral.textDisabled};
100
- `}
101
-
102
- ${({ readOnly, theme: { colors } }) => readOnly && css`
103
- color: ${colors.neutral.textDisabled};
104
- `}
105
-
106
- ${({ error, theme: { colors } }) => error && css`
107
- color: ${colors.danger.text};
108
- `}
109
- `;
110
- const StyledRelativeDiv = styled.div`
111
- position: relative;
112
- `;
113
- const StyledError = styled.div`
114
- font-size: 12px;
115
- color: ${({ theme }) => theme.colors.danger.text};
116
- padding-top: ${({ theme }) => theme.space["0.25"]};
117
- `;
118
- const StyledNotice = styled(Notice)`
119
- margin-top: ${({ theme }) => theme.space["0.5"]};
120
- `;
121
- const StyledInput = styled("input", {
122
- shouldForwardProp: (prop) => ![
123
- "as",
124
- "error",
125
- "fillAvailable",
126
- "hasLabel",
127
- "isPlaceholderVisible",
128
- "multiline",
129
- "resizable",
130
- "inputSize",
131
- "paddingRightFactor",
132
- "rightComponentLength",
133
- "unit"
134
- ].includes(prop)
135
- })`
136
- transition:
137
- border-color 0.2s ease,
138
- box-shadow 0.2s ease;
139
- appearance: none;
140
- background-color: ${({ theme: { colors } }) => colors.neutral.background};
141
- background-image: none;
142
- border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};
143
- border-radius: ${({ theme: { radii } }) => radii.default};
144
- color: ${({ theme: { colors } }) => colors.neutral.text};
145
- display: block;
146
- max-width: 100%;
147
- outline: none;
148
- position: relative;
149
- width: 100%;
150
- padding-left: ${({ theme }) => theme.space["1"]};
151
- padding-right: ${({ theme }) => theme.space["1"]};
152
- padding-top: 14px;
153
- font-size: 16px;
154
- line-height: 24px;
155
-
156
- &::placeholder {
157
- color: ${({ theme: { colors } }) => colors.neutral.textWeak};
158
- opacity: 0;
41
+ const StyledSeparator = /* @__PURE__ */ _styled(Separator, process.env.NODE_ENV === "production" ? {
42
+ target: "el3h3g98"
43
+ } : {
44
+ target: "el3h3g98",
45
+ label: "StyledSeparator"
46
+ })("margin:1px 0px;height:calc(100% - 2px);background-color:", ({
47
+ theme: {
48
+ colors
159
49
  }
160
-
161
- &:hover,
162
- &:focus {
163
- border-color: ${({ theme: { colors } }) => colors.primary.borderHover};
50
+ }) => colors.neutral.backgroundStrong, ";" + (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/TextInput/index.tsx"],"names":[],"mappings":"AA0DyC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"));
51
+ var _ref2 = process.env.NODE_ENV === "production" ? {
52
+ name: "176jwhy",
53
+ styles: "pointer-events:auto;>button{box-shadow:none!important;}"
54
+ } : {
55
+ name: "17s4cfn-StyledRightElement",
56
+ styles: "pointer-events:auto;>button{box-shadow:none!important;};label:StyledRightElement;",
57
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx"],"names":[],"mappings":"AA+FO","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */",
58
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
59
+ };
60
+ const StyledRightElement = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
61
+ shouldForwardProp: (prop) => !["edit", "touchable"].includes(prop),
62
+ target: "el3h3g97"
63
+ } : {
64
+ shouldForwardProp: (prop) => !["edit", "touchable"].includes(prop),
65
+ target: "el3h3g97",
66
+ label: "StyledRightElement"
67
+ })(({
68
+ theme: {
69
+ colors,
70
+ space
164
71
  }
165
-
166
- &:focus {
167
- box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};
168
- border-color: ${({ theme: { colors } }) => colors.primary.borderHover};
72
+ }) => /* @__PURE__ */ css("pointer-events:none;position:absolute;right:0;bottom:0;top:0;padding:0 ", space["1"], ";display:flex;gap:", space["1"], ";align-items:center;transition:transform 150ms,color 150ms;color:", colors.neutral.textWeak, ";&:hover,&:focus-within{color:", colors.neutral.textWeakHover, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledRightElement;"), 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/TextInput/index.tsx"],"names":[],"mappings":"AAwEyC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"), " ", ({
73
+ touchable
74
+ }) => touchable && _ref2, ";" + (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/TextInput/index.tsx"],"names":[],"mappings":"AAuE2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"));
75
+ var _ref = process.env.NODE_ENV === "production" ? {
76
+ name: "z8qxmc",
77
+ styles: "transform:translate(-9.6%, -3px) scale(0.8)"
78
+ } : {
79
+ name: "136ci60-StyledLabel",
80
+ styles: "transform:translate(-9.6%, -3px) scale(0.8);label:StyledLabel;",
81
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx"],"names":[],"mappings":"AAwIO","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */",
82
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
83
+ };
84
+ const StyledLabel = /* @__PURE__ */ _styled("label", process.env.NODE_ENV === "production" ? {
85
+ shouldForwardProp: (prop) => !["edit", "error", "resizable", "fillAvailable"].includes(prop),
86
+ target: "el3h3g96"
87
+ } : {
88
+ shouldForwardProp: (prop) => !["edit", "error", "resizable", "fillAvailable"].includes(prop),
89
+ target: "el3h3g96",
90
+ label: "StyledLabel"
91
+ })("display:block;position:absolute;left:0;top:0;padding-left:8px;padding-right:8px;pointer-events:none;color:", ({
92
+ theme: {
93
+ colors
169
94
  }
170
-
171
- ${({ isPlaceholderVisible }) => isPlaceholderVisible && `&::placeholder {
95
+ }) => colors.neutral.textWeak, ";white-space:nowrap;overflow:hidden;text-overflow:ellipsis;width:100%;height:48px;font-size:", ({
96
+ theme
97
+ }) => theme.typography.bodySmall, ";transition:transform 150ms;transform:translate(0, 12px) scale(1);", ({
98
+ edit
99
+ }) => edit && _ref, " ", ({
100
+ disabled,
101
+ theme: {
102
+ colors
103
+ }
104
+ }) => disabled && /* @__PURE__ */ css("color:", colors.neutral.textDisabled, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledLabel;"), 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/TextInput/index.tsx"],"names":[],"mappings":"AA8IO","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"), " ", ({
105
+ readOnly,
106
+ theme: {
107
+ colors
108
+ }
109
+ }) => readOnly && /* @__PURE__ */ css("color:", colors.neutral.textDisabled, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledLabel;"), 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/TextInput/index.tsx"],"names":[],"mappings":"AAoJO","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"), " ", ({
110
+ error,
111
+ theme: {
112
+ colors
113
+ }
114
+ }) => error && /* @__PURE__ */ css("color:", colors.danger.text, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:StyledLabel;"), 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/TextInput/index.tsx"],"names":[],"mappings":"AA0JO","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"), ";" + (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/TextInput/index.tsx"],"names":[],"mappings":"AAoHoB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"));
115
+ const StyledRelativeDiv = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
116
+ target: "el3h3g95"
117
+ } : {
118
+ target: "el3h3g95",
119
+ label: "StyledRelativeDiv"
120
+ })(process.env.NODE_ENV === "production" ? {
121
+ name: "bjn8wh",
122
+ styles: "position:relative"
123
+ } : {
124
+ name: "bjn8wh",
125
+ styles: "position:relative",
126
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx"],"names":[],"mappings":"AA+JoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */",
127
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
128
+ });
129
+ const StyledError = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
130
+ target: "el3h3g94"
131
+ } : {
132
+ target: "el3h3g94",
133
+ label: "StyledError"
134
+ })("font-size:12px;color:", ({
135
+ theme
136
+ }) => theme.colors.danger.text, ";padding-top:", ({
137
+ theme
138
+ }) => theme.space["0.25"], ";" + (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/TextInput/index.tsx"],"names":[],"mappings":"AAmK8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"));
139
+ const StyledNotice = /* @__PURE__ */ _styled(Notice, process.env.NODE_ENV === "production" ? {
140
+ target: "el3h3g93"
141
+ } : {
142
+ target: "el3h3g93",
143
+ label: "StyledNotice"
144
+ })("margin-top:", ({
145
+ theme
146
+ }) => 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/TextInput/index.tsx"],"names":[],"mappings":"AAyKmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"));
147
+ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
148
+ shouldForwardProp: (prop) => !["as", "error", "fillAvailable", "hasLabel", "isPlaceholderVisible", "multiline", "resizable", "inputSize", "paddingRightFactor", "rightComponentLength", "unit"].includes(prop),
149
+ target: "el3h3g92"
150
+ } : {
151
+ shouldForwardProp: (prop) => !["as", "error", "fillAvailable", "hasLabel", "isPlaceholderVisible", "multiline", "resizable", "inputSize", "paddingRightFactor", "rightComponentLength", "unit"].includes(prop),
152
+ target: "el3h3g92",
153
+ label: "StyledInput"
154
+ })("transition:border-color 0.2s ease,box-shadow 0.2s ease;appearance:none;background-color:", ({
155
+ theme: {
156
+ colors
157
+ }
158
+ }) => colors.neutral.background, ";background-image:none;border:1px solid ", ({
159
+ theme: {
160
+ colors
161
+ }
162
+ }) => colors.neutral.border, ";border-radius:", ({
163
+ theme: {
164
+ radii
165
+ }
166
+ }) => radii.default, ";color:", ({
167
+ theme: {
168
+ colors
169
+ }
170
+ }) => colors.neutral.text, ";display:block;max-width:100%;outline:none;position:relative;width:100%;padding-left:", ({
171
+ theme
172
+ }) => theme.space["1"], ";padding-right:", ({
173
+ theme
174
+ }) => theme.space["1"], ";padding-top:14px;font-size:16px;line-height:24px;&::placeholder{color:", ({
175
+ theme: {
176
+ colors
177
+ }
178
+ }) => colors.neutral.textWeak, ";opacity:0;}&:hover,&:focus{border-color:", ({
179
+ theme: {
180
+ colors
181
+ }
182
+ }) => colors.primary.borderHover, ";}&:focus{box-shadow:", ({
183
+ theme: {
184
+ shadows
185
+ }
186
+ }) => shadows.focusPrimary, ";border-color:", ({
187
+ theme: {
188
+ colors
189
+ }
190
+ }) => colors.primary.borderHover, ";}", ({
191
+ isPlaceholderVisible
192
+ }) => isPlaceholderVisible && `&::placeholder {
172
193
  opacity: 1;
173
- }`}
174
-
175
- ${({ disabled, theme: { colors } }) => disabled && `cursor: default;
194
+ }`, " ", ({
195
+ disabled,
196
+ theme: {
197
+ colors
198
+ }
199
+ }) => disabled && `cursor: default;
176
200
  pointer-events: none;
177
201
  background-color: ${colors.neutral.backgroundDisabled};
178
202
  border-color: ${colors.neutral.borderDisabled};
179
- color: ${colors.neutral.textDisabled};`}
180
-
181
- ${({ readOnly, theme: { colors } }) => readOnly && `background-color: ${colors.neutral.backgroundDisabled};
203
+ color: ${colors.neutral.textDisabled};`, " ", ({
204
+ readOnly,
205
+ theme: {
206
+ colors
207
+ }
208
+ }) => readOnly && `background-color: ${colors.neutral.backgroundDisabled};
182
209
  border-color: ${colors.neutral.borderDisabled};
183
- color: ${colors.neutral.text};`}
184
-
185
- ${({ inputSize }) => inputSizes[inputSize]?.default}
186
-
187
- ${({ inputSize, hasLabel }) => !!inputSize && !hasLabel && inputSizes[inputSize]?.full}
188
-
189
- ${({ error, theme: { colors, shadows } }) => error && `border-color: ${colors.danger.border};
210
+ color: ${colors.neutral.text};`, " ", ({
211
+ inputSize
212
+ }) => inputSizes[inputSize]?.default, " ", ({
213
+ inputSize,
214
+ hasLabel
215
+ }) => !!inputSize && !hasLabel && inputSizes[inputSize]?.full, " ", ({
216
+ error,
217
+ theme: {
218
+ colors,
219
+ shadows
220
+ }
221
+ }) => error && `border-color: ${colors.danger.border};
190
222
 
191
223
  &:hover,
192
224
  &:focus {
@@ -196,277 +228,160 @@ const StyledInput = styled("input", {
196
228
  &:focus {
197
229
  box-shadow: ${shadows.focusDanger};
198
230
  border-color: ${colors.danger.borderHover};
199
- }`}
200
-
201
- ${({ multiline, resizable, fillAvailable }) => multiline && `
231
+ }`, " ", ({
232
+ multiline,
233
+ resizable,
234
+ fillAvailable
235
+ }) => multiline && `
202
236
  padding-top: 20px;
203
237
  height: ${fillAvailable ? "100%" : "initial"};
204
238
  resize: ${resizable === false ? "none" : "vertical"};
205
- `}
206
-
207
- ${({ multiline, hasLabel, theme }) => multiline && !hasLabel && `
239
+ `, " ", ({
240
+ multiline,
241
+ hasLabel,
242
+ theme
243
+ }) => multiline && !hasLabel && `
208
244
  padding-top: ${theme.space["1"]};
209
- `}
210
-
211
- ${({ paddingRightFactor, rightComponentLength, unit, theme }) => paddingRightFactor > 0 && `
245
+ `, " ", ({
246
+ paddingRightFactor,
247
+ rightComponentLength,
248
+ unit,
249
+ theme
250
+ }) => paddingRightFactor > 0 && `
212
251
  padding-right: calc(${unit ? `${unit.length} * ${theme.space["1"]} + ` : ""}${paddingRightFactor + (unit ? rightComponentLength - 1 : rightComponentLength)} * ${theme.space["4"]});
213
- `}
214
- `;
215
- const RightComponent = styled(Stack)`
216
- min-width: 24px;
217
- `;
218
- const UnitLabel = styled(Text)`
219
- padding: ${({ theme }) => theme.space["1"]} 0;
220
- line-height: 18px;
221
- `;
222
- const TextInput = forwardRef(
223
- ({
224
- "data-testid": dataTestId,
225
- ariaControls,
226
- autoComplete = "on",
227
- autoFocus,
228
- className,
229
- cols,
230
- defaultValue,
231
- disabled,
232
- edit: forceEdit,
233
- error,
234
- fillAvailable,
235
- generated,
236
- height,
237
- id,
238
- label,
239
- multiline,
240
- name,
241
- notice,
242
- noTopLabel = false,
243
- onBlur,
244
- onChange,
245
- onFocus,
246
- onKeyUp,
247
- onKeyDown,
248
- placeholder,
249
- random,
250
- readOnly,
251
- required,
252
- resizable,
253
- rows,
254
- size = "medium",
255
- tabIndex,
256
- type = "text",
257
- unit,
258
- valid,
259
- value,
260
- wrap,
261
- inputProps,
262
- min,
263
- max
264
- }, ref) => {
265
- const controlRef = useRef(null);
266
- const [visited, setVisited] = useState(false);
267
- const [passwordVisible, setPasswordVisible] = useState(false);
268
- const togglePasswordVisibility = useCallback(
269
- () => setPasswordVisible((x) => !x),
270
- []
271
- );
272
- useImperativeHandle(ref, () => controlRef, []);
273
- useEffect(() => {
274
- if (passwordVisible) {
275
- controlRef?.current?.focus();
276
- }
277
- }, [passwordVisible]);
278
- const handlePassVisibilityClick = useCallback(
279
- () => togglePasswordVisibility(),
280
- [togglePasswordVisibility]
281
- );
282
- const randomize = useCallback(
283
- () => onChange?.(randomName(random)),
284
- [onChange, random]
285
- );
286
- const handleClickRandomize = useCallback(() => randomize(), [randomize]);
287
- const handleFocus = useCallback(
288
- (event) => {
289
- if (!visited && !readOnly) {
290
- setVisited(true);
291
- }
292
- if (onFocus) {
293
- onFocus(event);
294
- }
295
- },
296
- [visited, readOnly, onFocus]
297
- );
298
- const handleChange = useCallback(
299
- (event) => onChange?.(event.target.value),
300
- [onChange]
301
- );
302
- const isPassToggleable = type === "toggleable-password";
303
- const hasLabel = !!label && !noTopLabel && size === "medium";
304
- const edit = hasLabel && (forceEdit || visited || value || error || generated);
305
- const isPlaceholderVisible = !hasLabel || !!edit;
306
- const hasRightElement = !!(valid !== void 0 || isPassToggleable || random || unit);
307
- const getType = () => {
308
- if (isPassToggleable) {
309
- return passwordVisible || generated ? "text" : "password";
310
- }
311
- return multiline ? void 0 : type;
312
- };
313
- const inputSize = size;
314
- const rightComponentsArray = useMemo(() => {
315
- const rightComponents = [];
316
- if (isPassToggleable && !generated) {
317
- rightComponents.push(
318
- /* @__PURE__ */ jsx(
319
- Button,
320
- {
321
- "data-testid": dataTestId ? `${dataTestId}-visibility-button` : void 0,
322
- "aria-label": passwordVisible ? "hide" : "show",
323
- onClick: handlePassVisibilityClick,
324
- variant: "ghost",
325
- sentiment: "neutral",
326
- icon: passwordVisible ? "eye-off" : "eye",
327
- size: "small"
328
- },
329
- "password-visible"
330
- )
331
- );
332
- }
333
- if (random) {
334
- rightComponents.push(
335
- /* @__PURE__ */ jsx(
336
- Button,
337
- {
338
- "data-testid": dataTestId ? `${dataTestId}-randomize-button` : void 0,
339
- "aria-label": "randomize",
340
- onClick: handleClickRandomize,
341
- disabled,
342
- icon: "auto-fix",
343
- variant: "ghost",
344
- sentiment: "neutral",
345
- size: "small"
346
- },
347
- "random"
348
- )
349
- );
350
- }
351
- if (valid === false || valid === true) {
352
- rightComponents.push(
353
- /* @__PURE__ */ jsx(
354
- Icon,
355
- {
356
- name: !valid ? "close" : "check",
357
- color: !valid ? "danger" : "success",
358
- size: 20
359
- },
360
- "valid"
361
- )
362
- );
363
- }
364
- if (unit) {
365
- rightComponents.push(
366
- /* @__PURE__ */ jsx(UnitLabel, { variant: "bodySmall", as: "p", prominence: "weak", children: unit }, "unit")
367
- );
368
- }
369
- return rightComponents;
370
- }, [
371
- disabled,
372
- generated,
373
- handleClickRandomize,
374
- handlePassVisibilityClick,
375
- isPassToggleable,
376
- passwordVisible,
377
- random,
378
- unit,
379
- valid,
380
- dataTestId
381
- ]);
382
- const showSeparator = required && hasRightElement || unit;
383
- const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0);
384
- return /* @__PURE__ */ jsxs("div", { className, children: [
385
- /* @__PURE__ */ jsxs(StyledRelativeDiv, { children: [
386
- /* @__PURE__ */ jsx(
387
- StyledInput,
388
- {
389
- "aria-controls": ariaControls,
390
- "aria-label": label || void 0,
391
- "aria-labelledby": hasLabel ? ariaControls : void 0,
392
- as: multiline ? "textarea" : "input",
393
- autoComplete,
394
- autoFocus,
395
- cols,
396
- "data-testid": dataTestId,
397
- defaultValue,
398
- disabled,
399
- error: !!error,
400
- fillAvailable,
401
- hasLabel,
402
- paddingRightFactor,
403
- rightComponentLength: rightComponentsArray.length,
404
- unit,
405
- id,
406
- inputSize,
407
- isPlaceholderVisible,
408
- multiline,
409
- name,
410
- onBlur,
411
- onChange: handleChange,
412
- onFocus: handleFocus,
413
- onKeyUp,
414
- onKeyDown,
415
- placeholder,
416
- readOnly,
417
- ref: controlRef,
418
- resizable,
419
- rows,
420
- style: { height },
421
- tabIndex,
422
- type: getType(),
423
- value: value === null ? "" : value,
424
- wrap,
425
- min,
426
- max,
427
- required,
428
- ...inputProps
429
- }
430
- ),
431
- hasLabel && /* @__PURE__ */ jsx(
432
- StyledLabel,
433
- {
434
- edit: !!edit,
435
- disabled,
436
- readOnly,
437
- error: !!error,
438
- id: ariaControls,
439
- htmlFor: id,
440
- "aria-live": "assertive",
441
- children: label
442
- }
443
- ),
444
- hasRightElement || required ? /* @__PURE__ */ jsxs(
445
- StyledRightElement,
446
- {
447
- edit: !!edit,
448
- touchable: isPassToggleable || !!random,
449
- children: [
450
- required ? /* @__PURE__ */ jsx(Icon, { name: "asterisk", color: "danger", size: 10 }) : null,
451
- showSeparator ? /* @__PURE__ */ jsx(StyledSeparator, { direction: "vertical" }) : null,
452
- rightComponentsArray.length > 0 ? /* @__PURE__ */ jsx(
453
- RightComponent,
454
- {
455
- justifyContent: "center",
456
- direction: "row",
457
- alignItems: "center",
458
- children: rightComponentsArray
459
- }
460
- ) : null
461
- ]
462
- }
463
- ) : null
464
- ] }),
465
- /* @__PURE__ */ jsx(Expandable, { opened: !!error, children: /* @__PURE__ */ jsx(StyledError, { children: error }) }),
466
- notice ? /* @__PURE__ */ jsx(StyledNotice, { children: notice }) : null
467
- ] });
468
- }
469
- );
252
+ `, ";" + (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/TextInput/index.tsx"],"names":[],"mappings":"AAkNoB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"));
253
+ const RightComponent = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
254
+ target: "el3h3g91"
255
+ } : {
256
+ target: "el3h3g91",
257
+ label: "RightComponent"
258
+ })(process.env.NODE_ENV === "production" ? {
259
+ name: "9bmev",
260
+ styles: "min-width:24px"
261
+ } : {
262
+ name: "9bmev",
263
+ styles: "min-width:24px",
264
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx"],"names":[],"mappings":"AAwToC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */",
265
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
266
+ });
267
+ const UnitLabel = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
268
+ target: "el3h3g90"
269
+ } : {
270
+ target: "el3h3g90",
271
+ label: "UnitLabel"
272
+ })("padding:", ({
273
+ theme
274
+ }) => theme.space["1"], " 0;line-height:18px;" + (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/TextInput/index.tsx"],"names":[],"mappings":"AA0W8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextInput/index.tsx","sourcesContent":["import { css } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport randomName from '@scaleway/random-name'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  FocusEventHandler,\n  InputHTMLAttributes,\n  KeyboardEventHandler,\n  LabelHTMLAttributes,\n  TextareaHTMLAttributes,\n} from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Button } from '../Button'\nimport { Expandable } from '../Expandable'\nimport { Notice } from '../Notice'\nimport { Separator } from '../Separator'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\nconst inputSizes = {\n  medium: {\n    default: `\n      height: 48px;\n      padding-left: 8px;\n      padding-right: 20px;\n      padding-top: 14px;\n    `,\n    full: `\n      padding: 8px;\n    `,\n  },\n  small: {\n    default: `\n      height: 30px;\n      padding-left: 8px;\n      padding-right: 8px;\n      padding-top: 14px;\n      font-size: 14px;\n    `,\n    full: `\n      padding: 4px 8px;\n    `,\n  },\n}\n\ntype TextInputSizes = keyof typeof inputSizes\n\nexport const textInputSizes = Object.keys(inputSizes) as TextInputSizes[]\n\nconst StyledSeparator = styled(Separator)`\n  margin: 1px 0px;\n  height: calc(100% - 2px);\n  background-color: ${({ theme: { colors } }) =>\n    colors.neutral.backgroundStrong};\n`\ntype StyledRightElementProps = {\n  edit?: boolean\n  touchable?: boolean\n}\n\nconst StyledRightElement = styled('div', {\n  shouldForwardProp: prop => !['edit', 'touchable'].includes(prop),\n})<StyledRightElementProps>`\n  ${({ theme: { colors, space } }) => css`\n    pointer-events: none;\n    position: absolute;\n    right: 0;\n    bottom: 0;\n    top: 0;\n    padding: 0 ${space['1']};\n    display: flex;\n    gap: ${space['1']};\n    align-items: center;\n    transition:\n      transform 150ms,\n      color 150ms;\n    color: ${colors.neutral.textWeak};\n\n    &:hover,\n    &:focus-within {\n      color: ${colors.neutral.textWeakHover};\n    }\n  `}\n\n  ${({ touchable }) =>\n    touchable &&\n    css`\n      pointer-events: auto;\n      > button {\n        box-shadow: none !important;\n      }\n    `}\n`\ntype StyledLabelProps = {\n  'aria-label'?: string\n  'aria-live': string\n  disabled?: boolean\n  edit?: boolean\n  error?: boolean\n  readOnly?: boolean\n  resizable?: boolean\n  fillAvailable?: boolean\n} & LabelHTMLAttributes<HTMLLabelElement>\n\nconst StyledLabel = styled('label', {\n  shouldForwardProp: prop =>\n    !['edit', 'error', 'resizable', 'fillAvailable'].includes(prop),\n})<StyledLabelProps>`\n  display: block;\n  position: absolute;\n  left: 0;\n  top: 0;\n  padding-left: 8px;\n  padding-right: 8px;\n  pointer-events: none;\n  color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  width: 100%;\n  height: 48px;\n  font-size: ${({ theme }) => theme.typography.bodySmall};\n  transition: transform 150ms;\n  transform: translate(0, 12px) scale(1);\n\n  ${({ edit }) =>\n    edit &&\n    css`\n      transform: translate(-9.6%, -3px) scale(0.8);\n    `}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    css`\n      color: ${colors.neutral.textDisabled};\n    `}\n\n  ${({ error, theme: { colors } }) =>\n    error &&\n    css`\n      color: ${colors.danger.text};\n    `}\n`\n\nconst StyledRelativeDiv = styled.div`\n  position: relative;\n`\n\nconst StyledError = styled.div`\n  font-size: 12px;\n  color: ${({ theme }) => theme.colors.danger.text};\n  padding-top: ${({ theme }) => theme.space['0.25']};\n`\n\nconst StyledNotice = styled(Notice)`\n  margin-top: ${({ theme }) => theme.space['0.5']};\n`\n\ntype StyledInputProps = {\n  disabled?: boolean\n  error?: boolean\n  fillAvailable?: boolean\n  hasLabel?: boolean\n  paddingRightFactor: number\n  isPlaceholderVisible?: boolean\n  multiline?: boolean\n  resizable?: boolean\n  inputSize: TextInputSizes\n  unit?: string\n  rightComponentLength: number\n} & (\n  | InputHTMLAttributes<HTMLInputElement>\n  | TextareaHTMLAttributes<HTMLTextAreaElement>\n)\n\ntype InputProps = Omit<\n  Exclude<StyledInputProps, TextareaHTMLAttributes<HTMLTextAreaElement>>,\n  'inputSize'\n>\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop =>\n    ![\n      'as',\n      'error',\n      'fillAvailable',\n      'hasLabel',\n      'isPlaceholderVisible',\n      'multiline',\n      'resizable',\n      'inputSize',\n      'paddingRightFactor',\n      'rightComponentLength',\n      'unit',\n    ].includes(prop),\n})<StyledInputProps>`\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n  appearance: none;\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n  background-image: none;\n  border: 1px solid ${({ theme: { colors } }) => colors.neutral.border};\n  border-radius: ${({ theme: { radii } }) => radii.default};\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  display: block;\n  max-width: 100%;\n  outline: none;\n  position: relative;\n  width: 100%;\n  padding-left: ${({ theme }) => theme.space['1']};\n  padding-right: ${({ theme }) => theme.space['1']};\n  padding-top: 14px;\n  font-size: 16px;\n  line-height: 24px;\n\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n    opacity: 0;\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n    border-color: ${({ theme: { colors } }) => colors.primary.borderHover};\n  }\n\n  ${({ isPlaceholderVisible }) =>\n    isPlaceholderVisible &&\n    `&::placeholder {\n      opacity: 1;\n    }`}\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    pointer-events: none;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};`}\n\n  ${({ readOnly, theme: { colors } }) =>\n    readOnly &&\n    `background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.text};`}\n\n  ${({ inputSize }) => inputSizes[inputSize]?.default}\n\n  ${({ inputSize, hasLabel }) =>\n    !!inputSize && !hasLabel && inputSizes[inputSize]?.full}\n\n  ${({ error, theme: { colors, shadows } }) =>\n    error &&\n    `border-color: ${colors.danger.border};\n\n    &:hover,\n    &:focus {\n      border-color: ${colors.danger.borderHover};\n    }\n\n    &:focus {\n      box-shadow: ${shadows.focusDanger};\n      border-color: ${colors.danger.borderHover};\n    }`}\n\n    ${({ multiline, resizable, fillAvailable }) =>\n    multiline &&\n    `\n    padding-top: 20px;\n    height: ${fillAvailable ? '100%' : 'initial'};\n    resize: ${resizable === false ? 'none' : 'vertical'};\n  `}\n\n  ${({ multiline, hasLabel, theme }) =>\n    multiline &&\n    !hasLabel &&\n    `\n    padding-top: ${theme.space['1']};\n  `}\n\n  ${({ paddingRightFactor, rightComponentLength, unit, theme }) =>\n    paddingRightFactor > 0 &&\n    `\n    padding-right: calc(${\n      unit ? `${unit.length} * ${theme.space['1']} + ` : ''\n    }${\n      paddingRightFactor +\n      (unit ? rightComponentLength - 1 : rightComponentLength)\n    } * ${theme.space['4']});\n  `}\n`\n\nconst RightComponent = styled(Stack)`\n  min-width: 24px;\n`\n\ntype TextInputProps = {\n  'data-testid'?: string\n  ariaControls?: string\n  autoComplete?: string\n  autoFocus?: boolean\n  className?: string\n  cols?: number\n  defaultValue?: string\n  disabled?: boolean\n  edit?: boolean\n  error?: string\n  fillAvailable?: boolean\n  generated?: boolean\n  height?: string | number\n  id?: string\n  label?: string\n  multiline?: boolean\n  name?: string\n  notice?: string\n  noTopLabel?: boolean\n  onBlur?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onChange?: (value: string) => void\n  onFocus?: FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyUp?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  onKeyDown?: KeyboardEventHandler<HTMLInputElement | HTMLTextAreaElement>\n  placeholder?: string\n  random?: string\n  readOnly?: boolean\n  required?: boolean\n  resizable?: boolean\n  rows?: number\n  size?: TextInputSizes\n  tabIndex?: number\n  type?: string\n  unit?: string\n  valid?: boolean\n  value?: string | number\n  wrap?: string\n  inputProps?: InputProps\n  max?: InputHTMLAttributes<HTMLInputElement>['max']\n  min?: InputHTMLAttributes<HTMLInputElement>['min']\n} & (\n  | Omit<InputHTMLAttributes<HTMLInputElement>, 'onChange'>\n  | Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange'>\n)\n\nconst UnitLabel = styled(Text)`\n  padding: ${({ theme }) => theme.space['1']} 0;\n  line-height: 18px;\n`\n\n/**\n * TextInput component allows users to input text, with options for customization and validation.\n * It supports various input types and should be appropriately sized with clear labeling.\n */\nexport const TextInput = forwardRef<\n  HTMLInputElement | HTMLTextAreaElement | null,\n  TextInputProps\n>(\n  (\n    {\n      'data-testid': dataTestId,\n      ariaControls,\n      autoComplete = 'on',\n      autoFocus,\n      className,\n      cols,\n      defaultValue,\n      disabled,\n      edit: forceEdit,\n      error,\n      fillAvailable,\n      generated,\n      height,\n      id,\n      label,\n      multiline,\n      name,\n      notice,\n      noTopLabel = false,\n      onBlur,\n      onChange,\n      onFocus,\n      onKeyUp,\n      onKeyDown,\n      placeholder,\n      random,\n      readOnly,\n      required,\n      resizable,\n      rows,\n      size = 'medium',\n      tabIndex,\n      type = 'text',\n      unit,\n      valid,\n      value,\n      wrap,\n      inputProps,\n      min,\n      max,\n    },\n    ref,\n  ) => {\n    const controlRef = useRef<HTMLInputElement>(null)\n\n    const [visited, setVisited] = useState(false)\n    const [passwordVisible, setPasswordVisible] = useState(false)\n    const togglePasswordVisibility = useCallback(\n      () => setPasswordVisible(x => !x),\n      [],\n    )\n\n    // Forward ref to parent ref\n    useImperativeHandle<unknown, unknown>(ref, () => controlRef, [])\n\n    // Focus when password is visible\n    useEffect(() => {\n      if (passwordVisible) {\n        controlRef?.current?.focus()\n      }\n    }, [passwordVisible])\n\n    const handlePassVisibilityClick = useCallback(\n      () => togglePasswordVisibility(),\n      [togglePasswordVisibility],\n    )\n\n    const randomize = useCallback(\n      () => onChange?.(randomName(random)),\n      [onChange, random],\n    )\n\n    const handleClickRandomize = useCallback(() => randomize(), [randomize])\n\n    const handleFocus: FocusEventHandler<\n      HTMLInputElement | HTMLTextAreaElement\n    > = useCallback(\n      event => {\n        if (!visited && !readOnly) {\n          setVisited(true)\n        }\n\n        if (onFocus) {\n          onFocus(event)\n        }\n      },\n      [visited, readOnly, onFocus],\n    )\n\n    const handleChange = useCallback(\n      (event: ChangeEvent<HTMLInputElement | HTMLTextAreaElement>) =>\n        onChange?.(event.target.value),\n      [onChange],\n    )\n\n    const isPassToggleable = type === 'toggleable-password'\n    const hasLabel = !!label && !noTopLabel && size === 'medium'\n    const edit =\n      hasLabel && (forceEdit || visited || value || error || generated)\n\n    const isPlaceholderVisible = !hasLabel || !!edit\n    const hasRightElement = !!(\n      valid !== undefined ||\n      isPassToggleable ||\n      random ||\n      unit\n    )\n\n    const getType = () => {\n      if (isPassToggleable) {\n        return passwordVisible || generated ? 'text' : 'password'\n      }\n\n      return multiline ? undefined : type\n    }\n\n    const inputSize = size\n\n    const rightComponentsArray = useMemo(() => {\n      const rightComponents = []\n\n      if (isPassToggleable && !generated) {\n        rightComponents.push(\n          <Button\n            data-testid={\n              dataTestId ? `${dataTestId}-visibility-button` : undefined\n            }\n            aria-label={passwordVisible ? 'hide' : 'show'}\n            key=\"password-visible\"\n            onClick={handlePassVisibilityClick}\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            icon={passwordVisible ? 'eye-off' : 'eye'}\n            size=\"small\"\n          />,\n        )\n      }\n      if (random) {\n        rightComponents.push(\n          <Button\n            key=\"random\"\n            data-testid={\n              dataTestId ? `${dataTestId}-randomize-button` : undefined\n            }\n            aria-label=\"randomize\"\n            onClick={handleClickRandomize}\n            disabled={disabled}\n            icon=\"auto-fix\"\n            variant=\"ghost\"\n            sentiment=\"neutral\"\n            size=\"small\"\n          />,\n        )\n      }\n      if (valid === false || valid === true) {\n        rightComponents.push(\n          <Icon\n            key=\"valid\"\n            name={!valid ? 'close' : 'check'}\n            color={!valid ? 'danger' : 'success'}\n            size={20}\n          />,\n        )\n      }\n      if (unit) {\n        rightComponents.push(\n          <UnitLabel key=\"unit\" variant=\"bodySmall\" as=\"p\" prominence=\"weak\">\n            {unit}\n          </UnitLabel>,\n        )\n      }\n\n      return rightComponents\n    }, [\n      disabled,\n      generated,\n      handleClickRandomize,\n      handlePassVisibilityClick,\n      isPassToggleable,\n      passwordVisible,\n      random,\n      unit,\n      valid,\n      dataTestId,\n    ])\n\n    const showSeparator = (required && hasRightElement) || unit\n    const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0)\n\n    return (\n      <div className={className}>\n        <StyledRelativeDiv>\n          <StyledInput\n            aria-controls={ariaControls}\n            aria-label={label || undefined}\n            aria-labelledby={hasLabel ? ariaControls : undefined}\n            as={multiline ? 'textarea' : 'input'}\n            autoComplete={autoComplete}\n            autoFocus={autoFocus}\n            cols={cols}\n            data-testid={dataTestId}\n            defaultValue={defaultValue}\n            disabled={disabled}\n            error={!!error}\n            fillAvailable={fillAvailable}\n            hasLabel={hasLabel}\n            paddingRightFactor={paddingRightFactor}\n            rightComponentLength={rightComponentsArray.length}\n            unit={unit}\n            id={id}\n            inputSize={inputSize}\n            isPlaceholderVisible={isPlaceholderVisible}\n            multiline={multiline}\n            name={name}\n            onBlur={onBlur}\n            onChange={handleChange}\n            onFocus={handleFocus}\n            onKeyUp={onKeyUp}\n            onKeyDown={onKeyDown}\n            placeholder={placeholder}\n            readOnly={readOnly}\n            ref={controlRef}\n            resizable={resizable}\n            rows={rows}\n            style={{ height }}\n            tabIndex={tabIndex}\n            type={getType()}\n            value={value === null ? '' : value}\n            wrap={wrap}\n            min={min}\n            max={max}\n            required={required}\n            {...inputProps}\n          />\n          {hasLabel && (\n            <StyledLabel\n              edit={!!edit}\n              disabled={disabled}\n              readOnly={readOnly}\n              error={!!error}\n              id={ariaControls}\n              htmlFor={id}\n              aria-live=\"assertive\"\n            >\n              {label}\n            </StyledLabel>\n          )}\n\n          {hasRightElement || required ? (\n            <StyledRightElement\n              edit={!!edit}\n              touchable={isPassToggleable || !!random}\n            >\n              {required ? (\n                <Icon name=\"asterisk\" color=\"danger\" size={10} />\n              ) : null}\n              {showSeparator ? <StyledSeparator direction=\"vertical\" /> : null}\n              {rightComponentsArray.length > 0 ? (\n                <RightComponent\n                  justifyContent=\"center\"\n                  direction=\"row\"\n                  alignItems=\"center\"\n                >\n                  {rightComponentsArray}\n                </RightComponent>\n              ) : null}\n            </StyledRightElement>\n          ) : null}\n        </StyledRelativeDiv>\n        <Expandable opened={!!error}>\n          <StyledError>{error}</StyledError>\n        </Expandable>\n        {notice ? <StyledNotice>{notice}</StyledNotice> : null}\n      </div>\n    )\n  },\n)\n"]} */"));
275
+ const TextInput = forwardRef(({
276
+ "data-testid": dataTestId,
277
+ ariaControls,
278
+ autoComplete = "on",
279
+ autoFocus,
280
+ className,
281
+ cols,
282
+ defaultValue,
283
+ disabled,
284
+ edit: forceEdit,
285
+ error,
286
+ fillAvailable,
287
+ generated,
288
+ height,
289
+ id,
290
+ label,
291
+ multiline,
292
+ name,
293
+ notice,
294
+ noTopLabel = false,
295
+ onBlur,
296
+ onChange,
297
+ onFocus,
298
+ onKeyUp,
299
+ onKeyDown,
300
+ placeholder,
301
+ random,
302
+ readOnly,
303
+ required,
304
+ resizable,
305
+ rows,
306
+ size = "medium",
307
+ tabIndex,
308
+ type = "text",
309
+ unit,
310
+ valid,
311
+ value,
312
+ wrap,
313
+ inputProps,
314
+ min,
315
+ max
316
+ }, ref) => {
317
+ const controlRef = useRef(null);
318
+ const [visited, setVisited] = useState(false);
319
+ const [passwordVisible, setPasswordVisible] = useState(false);
320
+ const togglePasswordVisibility = useCallback(() => setPasswordVisible((x) => !x), []);
321
+ useImperativeHandle(ref, () => controlRef, []);
322
+ useEffect(() => {
323
+ if (passwordVisible) {
324
+ controlRef?.current?.focus();
325
+ }
326
+ }, [passwordVisible]);
327
+ const handlePassVisibilityClick = useCallback(() => togglePasswordVisibility(), [togglePasswordVisibility]);
328
+ const randomize = useCallback(() => onChange?.(randomName(random)), [onChange, random]);
329
+ const handleClickRandomize = useCallback(() => randomize(), [randomize]);
330
+ const handleFocus = useCallback((event) => {
331
+ if (!visited && !readOnly) {
332
+ setVisited(true);
333
+ }
334
+ if (onFocus) {
335
+ onFocus(event);
336
+ }
337
+ }, [visited, readOnly, onFocus]);
338
+ const handleChange = useCallback((event) => onChange?.(event.target.value), [onChange]);
339
+ const isPassToggleable = type === "toggleable-password";
340
+ const hasLabel = !!label && !noTopLabel && size === "medium";
341
+ const edit = hasLabel && (forceEdit || visited || value || error || generated);
342
+ const isPlaceholderVisible = !hasLabel || !!edit;
343
+ const hasRightElement = !!(valid !== void 0 || isPassToggleable || random || unit);
344
+ const getType = () => {
345
+ if (isPassToggleable) {
346
+ return passwordVisible || generated ? "text" : "password";
347
+ }
348
+ return multiline ? void 0 : type;
349
+ };
350
+ const inputSize = size;
351
+ const rightComponentsArray = useMemo(() => {
352
+ const rightComponents = [];
353
+ if (isPassToggleable && !generated) {
354
+ rightComponents.push(/* @__PURE__ */ jsx(Button, { "data-testid": dataTestId ? `${dataTestId}-visibility-button` : void 0, "aria-label": passwordVisible ? "hide" : "show", onClick: handlePassVisibilityClick, variant: "ghost", sentiment: "neutral", icon: passwordVisible ? "eye-off" : "eye", size: "small" }, "password-visible"));
355
+ }
356
+ if (random) {
357
+ rightComponents.push(/* @__PURE__ */ jsx(Button, { "data-testid": dataTestId ? `${dataTestId}-randomize-button` : void 0, "aria-label": "randomize", onClick: handleClickRandomize, disabled, icon: "auto-fix", variant: "ghost", sentiment: "neutral", size: "small" }, "random"));
358
+ }
359
+ if (valid === false || valid === true) {
360
+ rightComponents.push(/* @__PURE__ */ jsx(Icon, { name: !valid ? "close" : "check", color: !valid ? "danger" : "success", size: 20 }, "valid"));
361
+ }
362
+ if (unit) {
363
+ rightComponents.push(/* @__PURE__ */ jsx(UnitLabel, { variant: "bodySmall", as: "p", prominence: "weak", children: unit }, "unit"));
364
+ }
365
+ return rightComponents;
366
+ }, [disabled, generated, handleClickRandomize, handlePassVisibilityClick, isPassToggleable, passwordVisible, random, unit, valid, dataTestId]);
367
+ const showSeparator = required && hasRightElement || unit;
368
+ const paddingRightFactor = (required ? 1 : 0) + (showSeparator ? 0.5 : 0);
369
+ return /* @__PURE__ */ jsxs("div", { className, children: [
370
+ /* @__PURE__ */ jsxs(StyledRelativeDiv, { children: [
371
+ /* @__PURE__ */ jsx(StyledInput, { "aria-controls": ariaControls, "aria-label": label || void 0, "aria-labelledby": hasLabel ? ariaControls : void 0, as: multiline ? "textarea" : "input", autoComplete, autoFocus, cols, "data-testid": dataTestId, defaultValue, disabled, error: !!error, fillAvailable, hasLabel, paddingRightFactor, rightComponentLength: rightComponentsArray.length, unit, id, inputSize, isPlaceholderVisible, multiline, name, onBlur, onChange: handleChange, onFocus: handleFocus, onKeyUp, onKeyDown, placeholder, readOnly, ref: controlRef, resizable, rows, style: {
372
+ height
373
+ }, tabIndex, type: getType(), value: value === null ? "" : value, wrap, min, max, required, ...inputProps }),
374
+ hasLabel && /* @__PURE__ */ jsx(StyledLabel, { edit: !!edit, disabled, readOnly, error: !!error, id: ariaControls, htmlFor: id, "aria-live": "assertive", children: label }),
375
+ hasRightElement || required ? /* @__PURE__ */ jsxs(StyledRightElement, { edit: !!edit, touchable: isPassToggleable || !!random, children: [
376
+ required ? /* @__PURE__ */ jsx(Icon, { name: "asterisk", color: "danger", size: 10 }) : null,
377
+ showSeparator ? /* @__PURE__ */ jsx(StyledSeparator, { direction: "vertical" }) : null,
378
+ rightComponentsArray.length > 0 ? /* @__PURE__ */ jsx(RightComponent, { justifyContent: "center", direction: "row", alignItems: "center", children: rightComponentsArray }) : null
379
+ ] }) : null
380
+ ] }),
381
+ /* @__PURE__ */ jsx(Expandable, { opened: !!error, children: /* @__PURE__ */ jsx(StyledError, { children: error }) }),
382
+ notice ? /* @__PURE__ */ jsx(StyledNotice, { children: notice }) : null
383
+ ] });
384
+ });
470
385
  export {
471
386
  TextInput
472
387
  };