@ultraviolet/ui 1.86.0 → 1.87.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/dist/components/Chip/index.d.ts +1 -1
  2. package/dist/components/Label/index.cjs +23 -0
  3. package/dist/components/Label/index.d.ts +14 -0
  4. package/dist/components/Label/index.js +23 -0
  5. package/dist/components/MenuV2/components/Item.cjs +6 -4
  6. package/dist/components/MenuV2/components/Item.d.ts +3 -1
  7. package/dist/components/MenuV2/components/Item.js +6 -4
  8. package/dist/components/MenuV2/index.d.ts +3 -1
  9. package/dist/components/NumberInputV2/index.cjs +12 -18
  10. package/dist/components/NumberInputV2/index.js +7 -13
  11. package/dist/components/Popup/helpers.cjs +2 -2
  12. package/dist/components/Popup/helpers.js +2 -2
  13. package/dist/components/ProgressBar/index.cjs +8 -15
  14. package/dist/components/ProgressBar/index.js +8 -15
  15. package/dist/components/SelectInputV2/index.cjs +4 -8
  16. package/dist/components/SelectInputV2/index.js +4 -8
  17. package/dist/components/Tabs/TabMenuItem.d.ts +3 -1
  18. package/dist/components/Tabs/index.d.ts +3 -1
  19. package/dist/components/TagInput/index.cjs +8 -13
  20. package/dist/components/TagInput/index.js +7 -12
  21. package/dist/components/TextArea/index.cjs +8 -13
  22. package/dist/components/TextArea/index.js +6 -11
  23. package/dist/components/TextInputV2/index.cjs +18 -23
  24. package/dist/components/TextInputV2/index.js +9 -14
  25. package/dist/components/TimeInputV2/index.cjs +14 -32
  26. package/dist/components/TimeInputV2/index.d.ts +7 -2
  27. package/dist/components/TimeInputV2/index.js +14 -32
  28. package/dist/components/UnitInput/index.cjs +15 -14
  29. package/dist/components/UnitInput/index.js +14 -13
  30. package/dist/components/VerificationCode/index.cjs +5 -12
  31. package/dist/components/VerificationCode/index.js +5 -12
  32. package/dist/components/index.d.ts +1 -0
  33. package/dist/index.cjs +112 -110
  34. package/dist/index.js +2 -0
  35. package/package.json +2 -2
@@ -7,9 +7,10 @@ const Icon = require("@ultraviolet/icons");
7
7
  const React = require("react");
8
8
  const index$3 = require("../Button/index.cjs");
9
9
  const constants = require("../Button/constants.cjs");
10
+ const index$1 = require("../Label/index.cjs");
10
11
  const index$4 = require("../Row/index.cjs");
11
12
  const index = require("../Stack/index.cjs");
12
- const index$1 = require("../Text/index.cjs");
13
+ const index$5 = require("../Text/index.cjs");
13
14
  const index$2 = require("../Tooltip/index.cjs");
14
15
  const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
15
16
  const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
@@ -27,7 +28,7 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled__default.default("div", pr
27
28
  styles: "position:relative;display:flex"
28
29
  } : {
29
30
  name: "8k1832",
30
- styles: "position:relative;display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AAyBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
31
+ styles: "position:relative;display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AAsBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Label } from '../Label'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
31
32
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
32
33
  });
33
34
  const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
@@ -39,7 +40,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(ind
39
40
  theme
40
41
  }) => theme.space["1.5"], ";right:", ({
41
42
  theme
42
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA8BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
43
+ }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA2BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Label } from '../Label'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
43
44
  const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", process.env.NODE_ENV === "production" ? {
44
45
  shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
45
46
  target: "enu776d0"
@@ -88,7 +89,7 @@ const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", proc
88
89
  theme
89
90
  }) => theme.colors.primary.border, ";box-shadow:", ({
90
91
  theme
91
- }) => theme.shadows.focusPrimary, ";}}" + (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/TextArea/index.tsx"],"names":[],"mappings":"AA2CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
92
+ }) => theme.shadows.focusPrimary, ";}}" + (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/TextArea/index.tsx"],"names":[],"mappings":"AAwCuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Label } from '../Label'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
92
93
  const TextArea = React.forwardRef(({
93
94
  id,
94
95
  className,
@@ -153,13 +154,7 @@ const TextArea = React.forwardRef(({
153
154
  const notice = success || error || helper;
154
155
  const computedClearable = clearable && !!value;
155
156
  return /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: "0.5", className, children: [
156
- label || labelDescription ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "1", alignItems: "center", children: [
157
- label ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
158
- /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "label", variant: "bodyStrong", sentiment: "neutral", htmlFor: id ?? localId, prominence: "strong", children: label }),
159
- required ? /* @__PURE__ */ jsxRuntime.jsx(Icon.AsteriskIcon, { sentiment: "danger", size: 8 }) : null
160
- ] }) : null,
161
- labelDescription ?? null
162
- ] }) : null,
157
+ label || labelDescription ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Label, { labelDescription, required, htmlFor: id ?? localId, children: label }) : null,
163
158
  /* @__PURE__ */ jsxRuntime.jsx(index$2.Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsxs(StyledTextAreaWrapper, { children: [
164
159
  /* @__PURE__ */ jsxRuntime.jsx(StyledTextArea, { "aria-invalid": !!error, id: id ?? localId, tabIndex, autoFocus, disabled, rows: rows !== "auto" ? rows : 1, ref: textAreaRef, value, onChange: (event) => {
165
160
  onChange(event.currentTarget.value);
@@ -174,10 +169,10 @@ const TextArea = React.forwardRef(({
174
169
  ] }) }),
175
170
  notice || maxLength ? /* @__PURE__ */ jsxRuntime.jsxs(index$4.Row, { templateColumns: "minmax(0, 1fr) min-content", gap: "1", children: [
176
171
  /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
177
- error || success || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", variant: "caption", sentiment, prominence: !error && !success ? "weak" : "default", disabled, children: error || success || helper }) : null,
172
+ error || success || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$5.Text, { as: "p", variant: "caption", sentiment, prominence: !error && !success ? "weak" : "default", disabled, children: error || success || helper }) : null,
178
173
  !error && !success && typeof helper !== "string" && helper ? helper : null
179
174
  ] }),
180
- maxLength ? /* @__PURE__ */ jsxRuntime.jsxs(index$1.Text, { as: "div", sentiment: "neutral", prominence: "weak", variant: "caption", children: [
175
+ maxLength ? /* @__PURE__ */ jsxRuntime.jsxs(index$5.Text, { as: "div", sentiment: "neutral", prominence: "weak", variant: "caption", children: [
181
176
  value?.length ?? 0,
182
177
  "/",
183
178
  maxLength
@@ -1,10 +1,11 @@
1
1
  import { jsxs, jsx } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
3
  import { useTheme } from "@emotion/react";
4
- import { AsteriskIcon, CheckCircleIcon, AlertCircleIcon } from "@ultraviolet/icons";
4
+ import { CheckCircleIcon, AlertCircleIcon } from "@ultraviolet/icons";
5
5
  import { forwardRef, useId, useRef, useImperativeHandle, useEffect, useMemo } from "react";
6
6
  import { Button } from "../Button/index.js";
7
7
  import { SIZE_HEIGHT } from "../Button/constants.js";
8
+ import { Label } from "../Label/index.js";
8
9
  import { Row } from "../Row/index.js";
9
10
  import { Stack } from "../Stack/index.js";
10
11
  import { Text } from "../Text/index.js";
@@ -23,7 +24,7 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_EN
23
24
  styles: "position:relative;display:flex"
24
25
  } : {
25
26
  name: "8k1832",
26
- styles: "position:relative;display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AAyBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
27
+ styles: "position:relative;display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AAsBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Label } from '../Label'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
27
28
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
28
29
  });
29
30
  const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
@@ -35,7 +36,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.N
35
36
  theme
36
37
  }) => theme.space["1.5"], ";right:", ({
37
38
  theme
38
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA8BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
39
+ }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA2BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Label } from '../Label'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
39
40
  const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV === "production" ? {
40
41
  shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
41
42
  target: "enu776d0"
@@ -84,7 +85,7 @@ const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV
84
85
  theme
85
86
  }) => theme.colors.primary.border, ";box-shadow:", ({
86
87
  theme
87
- }) => theme.shadows.focusPrimary, ";}}" + (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/TextArea/index.tsx"],"names":[],"mappings":"AA2CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
88
+ }) => theme.shadows.focusPrimary, ";}}" + (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/TextArea/index.tsx"],"names":[],"mappings":"AAwCuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon } from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Label } from '../Label'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Label\n            labelDescription={labelDescription}\n            required={required}\n            htmlFor={id ?? localId}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
88
89
  const TextArea = forwardRef(({
89
90
  id,
90
91
  className,
@@ -149,13 +150,7 @@ const TextArea = forwardRef(({
149
150
  const notice = success || error || helper;
150
151
  const computedClearable = clearable && !!value;
151
152
  return /* @__PURE__ */ jsxs(Stack, { gap: "0.5", className, children: [
152
- label || labelDescription ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "1", alignItems: "center", children: [
153
- label ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
154
- /* @__PURE__ */ jsx(Text, { as: "label", variant: "bodyStrong", sentiment: "neutral", htmlFor: id ?? localId, prominence: "strong", children: label }),
155
- required ? /* @__PURE__ */ jsx(AsteriskIcon, { sentiment: "danger", size: 8 }) : null
156
- ] }) : null,
157
- labelDescription ?? null
158
- ] }) : null,
153
+ label || labelDescription ? /* @__PURE__ */ jsx(Label, { labelDescription, required, htmlFor: id ?? localId, children: label }) : null,
159
154
  /* @__PURE__ */ jsx(Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxs(StyledTextAreaWrapper, { children: [
160
155
  /* @__PURE__ */ jsx(StyledTextArea, { "aria-invalid": !!error, id: id ?? localId, tabIndex, autoFocus, disabled, rows: rows !== "auto" ? rows : 1, ref: textAreaRef, value, onChange: (event) => {
161
156
  onChange(event.currentTarget.value);