@ultraviolet/ui 3.0.0-beta.1 → 3.0.0-beta.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -25,11 +25,11 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled__default.default("div", pr
25
25
  target: "enu776d2",
26
26
  label: "StyledTextAreaWrapper"
27
27
  })(process.env.NODE_ENV === "production" ? {
28
- name: "8k1832",
29
- styles: "position:relative;display:flex"
28
+ name: "1kzcmo2",
29
+ styles: "position:relative;display:flex;width:100%"
30
30
  } : {
31
- name: "8k1832",
32
- 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":"AAwBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
31
+ name: "1kzcmo2",
32
+ styles: "position:relative;display:flex;width:100%/*# 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":"AAwBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n  width: 100%;\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
33
33
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
34
34
  });
35
35
  const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
@@ -41,7 +41,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(ind
41
41
  theme
42
42
  }) => theme.space["1.5"], ";right:", ({
43
43
  theme
44
- }) => 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":"AA6BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
44
+ }) => 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":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n  width: 100%;\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
45
45
  const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", process.env.NODE_ENV === "production" ? {
46
46
  shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
47
47
  target: "enu776d0"
@@ -90,7 +90,7 @@ const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", proc
90
90
  theme
91
91
  }) => theme.colors.primary.border, ";box-shadow:", ({
92
92
  theme
93
- }) => 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":"AA0CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
93
+ }) => 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":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n  width: 100%;\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
94
94
  const TextArea = react.forwardRef(({
95
95
  id,
96
96
  className,
@@ -21,11 +21,11 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_EN
21
21
  target: "enu776d2",
22
22
  label: "StyledTextAreaWrapper"
23
23
  })(process.env.NODE_ENV === "production" ? {
24
- name: "8k1832",
25
- styles: "position:relative;display:flex"
24
+ name: "1kzcmo2",
25
+ styles: "position:relative;display:flex;width:100%"
26
26
  } : {
27
- name: "8k1832",
28
- 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":"AAwBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
27
+ name: "1kzcmo2",
28
+ styles: "position:relative;display:flex;width:100%/*# 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":"AAwBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n  width: 100%;\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
29
29
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
30
30
  });
31
31
  const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
@@ -37,7 +37,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.N
37
37
  theme
38
38
  }) => theme.space["1.5"], ";right:", ({
39
39
  theme
40
- }) => 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":"AA6BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
40
+ }) => 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":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n  width: 100%;\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
41
41
  const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV === "production" ? {
42
42
  shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
43
43
  target: "enu776d0"
@@ -86,7 +86,7 @@ const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV
86
86
  theme
87
87
  }) => theme.colors.primary.border, ";box-shadow:", ({
88
88
  theme
89
- }) => 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":"AA0CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
89
+ }) => 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":["'use client'\n\nimport { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon, CheckCircleIcon, CloseIcon } 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 = 'small'\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n  width: 100%;\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, textAreaRef.current?.value])\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 className={className} gap=\"0.5\">\n        {label || labelDescription ? (\n          <Label\n            htmlFor={id ?? localId}\n            labelDescription={labelDescription}\n            required={required}\n          >\n            {label}\n          </Label>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              aria-label={ariaLabel}\n              autoFocus={autoFocus}\n              data-error={!!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-testid={dataTestId}\n              disabled={disabled}\n              hasSentimentIcon={!!success || !!error}\n              id={id ?? localId}\n              isClearable={!!computedClearable}\n              maxLength={maxLength}\n              minLength={minLength}\n              name={name}\n              onBlur={onBlur}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              onFocus={onFocus}\n              onKeyDown={onKeyDown}\n              placeholder={placeholder}\n              ref={textAreaRef}\n              rows={rows !== 'auto' ? rows : 1}\n              tabIndex={tabIndex}\n              value={value}\n            />\n            <StyledTextAreaAbsoluteStack\n              alignItems=\"center\"\n              direction=\"row\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                  size=\"xsmall\"\n                  variant=\"ghost\"\n                >\n                  <CloseIcon />\n                </Button>\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 gap=\"1\" templateColumns=\"minmax(0, 1fr) min-content\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  disabled={disabled}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  sentiment={sentiment}\n                  variant=\"caption\"\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                prominence=\"weak\"\n                sentiment=\"neutral\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
90
90
  const TextArea = forwardRef(({
91
91
  id,
92
92
  className,
@@ -36,6 +36,6 @@ export declare const Tooltip: import("react").ForwardRefExoticComponent<Pick<{
36
36
  disableAnimation?: boolean;
37
37
  portalTarget?: HTMLElement;
38
38
  dynamicDomRendering?: boolean;
39
- } & import("react").RefAttributes<HTMLDivElement>, "children" | "className" | "text" | "id" | "tabIndex" | "role" | "maxWidth" | "data-testid" | "containerFullHeight" | "containerFullWidth" | "visible" | "debounceDelay" | "portalTarget" | "innerRef"> & {
39
+ } & import("react").RefAttributes<HTMLDivElement>, "children" | "className" | "text" | "maxWidth" | "id" | "tabIndex" | "role" | "data-testid" | "containerFullHeight" | "containerFullWidth" | "visible" | "debounceDelay" | "portalTarget" | "innerRef"> & {
40
40
  placement?: Exclude<ComponentProps<typeof Popup>["placement"], "nested-menu">;
41
41
  } & import("react").RefAttributes<HTMLDivElement>>;
@@ -16,6 +16,28 @@ const useTheme = () => {
16
16
  const ThemeProvider = ({
17
17
  children,
18
18
  theme = themes.consoleLightTheme
19
- }) => /* @__PURE__ */ jsxRuntime.jsx(ThemeContext.Provider, { value: theme, children: /* @__PURE__ */ jsxRuntime.jsx("div", { style: dynamic.assignInlineVars(themes.theme, theme), children }) });
19
+ }) => {
20
+ react.useEffect(() => {
21
+ const styleId = "uv-theme";
22
+ const existingStyle = document.getElementById(styleId);
23
+ const cssVars = dynamic.assignInlineVars(themes.theme, theme);
24
+ const cssString = `:root { ${Object.entries(cssVars).map(([key, value]) => `${key}: ${value};`).join(" ")} }`;
25
+ if (existingStyle) {
26
+ existingStyle.textContent = cssString;
27
+ } else {
28
+ const style = document.createElement("style");
29
+ style.id = styleId;
30
+ style.textContent = cssString;
31
+ document.head.appendChild(style);
32
+ }
33
+ return () => {
34
+ const style = document.getElementById(styleId);
35
+ if (style) {
36
+ style.remove();
37
+ }
38
+ };
39
+ }, [theme]);
40
+ return /* @__PURE__ */ jsxRuntime.jsx(ThemeContext.Provider, { value: theme, children });
41
+ };
20
42
  exports.ThemeProvider = ThemeProvider;
21
43
  exports.useTheme = useTheme;
@@ -730,7 +730,7 @@ type ThemeProviderProps = {
730
730
  children: ReactNode;
731
731
  };
732
732
  /**
733
- * ThemeProvider will apply generated global CSS variables to the application.
733
+ * ThemeProvider will apply generated global CSS variables to the application in the `<head>`.
734
734
  * If no theme is provided, it will default to `lightTheme`.
735
735
  */
736
736
  export declare const ThemeProvider: ({ children, theme, }: ThemeProviderProps) => import("@emotion/react/jsx-runtime").JSX.Element;
@@ -1,8 +1,8 @@
1
1
  "use client";
2
2
  import { jsx } from "@emotion/react/jsx-runtime";
3
- import { consoleLightTheme, theme } from "@ultraviolet/themes";
3
+ import { theme, consoleLightTheme } from "@ultraviolet/themes";
4
4
  import { assignInlineVars } from "@vanilla-extract/dynamic";
5
- import { createContext, useContext } from "react";
5
+ import { useEffect, createContext, useContext } from "react";
6
6
  const ThemeContext = createContext(consoleLightTheme);
7
7
  const useTheme = () => {
8
8
  const context = useContext(ThemeContext);
@@ -14,7 +14,29 @@ const useTheme = () => {
14
14
  const ThemeProvider = ({
15
15
  children,
16
16
  theme: theme$1 = consoleLightTheme
17
- }) => /* @__PURE__ */ jsx(ThemeContext.Provider, { value: theme$1, children: /* @__PURE__ */ jsx("div", { style: assignInlineVars(theme, theme$1), children }) });
17
+ }) => {
18
+ useEffect(() => {
19
+ const styleId = "uv-theme";
20
+ const existingStyle = document.getElementById(styleId);
21
+ const cssVars = assignInlineVars(theme, theme$1);
22
+ const cssString = `:root { ${Object.entries(cssVars).map(([key, value]) => `${key}: ${value};`).join(" ")} }`;
23
+ if (existingStyle) {
24
+ existingStyle.textContent = cssString;
25
+ } else {
26
+ const style = document.createElement("style");
27
+ style.id = styleId;
28
+ style.textContent = cssString;
29
+ document.head.appendChild(style);
30
+ }
31
+ return () => {
32
+ const style = document.getElementById(styleId);
33
+ if (style) {
34
+ style.remove();
35
+ }
36
+ };
37
+ }, [theme$1]);
38
+ return /* @__PURE__ */ jsx(ThemeContext.Provider, { value: theme$1, children });
39
+ };
18
40
  export {
19
41
  ThemeProvider,
20
42
  useTheme
package/dist/ui.css CHANGED
@@ -1 +1 @@
1
- .uv_e1wcoe0{display:inline-flex;position:relative;flex-direction:row;box-sizing:border-box;align-items:center;justify-content:center;outline-offset:2px;white-space:nowrap;text-decoration:none;border-radius:var(--rwwhsl85)}.uv_e1wcoe0:hover{text-decoration:none}.uv_e1wcoe1{height:var(--rwwhsl9k);padding-left:var(--rwwhsl9v);padding-right:var(--rwwhsl9v);gap:var(--rwwhsl9u);font-size:var(--rwwhslb9);font-family:var(--rwwhslb8);font-weight:var(--rwwhslbg);letter-spacing:var(--rwwhslbb);line-height:var(--rwwhslbc)}.uv_e1wcoe2{height:var(--rwwhsl9j);padding-left:var(--rwwhsla6);padding-right:var(--rwwhsla6);gap:var(--rwwhsl9u);font-size:var(--rwwhslb9);font-family:var(--rwwhslb8);font-weight:var(--rwwhslbg);letter-spacing:var(--rwwhslbb);line-height:var(--rwwhslbc)}.uv_e1wcoe3{height:var(--rwwhsl9h);padding-left:var(--rwwhsl9u);padding-right:var(--rwwhsl9u);gap:var(--rwwhsl9u);font-size:var(--rwwhslb9);font-family:var(--rwwhslb8);font-weight:var(--rwwhslbg);letter-spacing:var(--rwwhslbb);line-height:var(--rwwhslbc)}.uv_e1wcoe4{height:var(--rwwhsl9f);padding-left:var(--rwwhsla4);padding-right:var(--rwwhsla4);gap:var(--rwwhsla4);font-size:var(--rwwhslb9);font-family:var(--rwwhslb8);font-weight:var(--rwwhslbg);letter-spacing:var(--rwwhslbb);line-height:var(--rwwhslbc)}.uv_e1wcoe5{width:100%}.uv_e1wcoe6{width:auto}.uv_e1wcoej{cursor:not-allowed}.uv_e1wcoek{cursor:pointer}.uv_e1wcoel{background:var(--rwwhsl5j);color:var(--rwwhsl61);border:none}.uv_e1wcoel:hover,.uv_e1wcoel:active{background:var(--rwwhsl5l);color:var(--rwwhsl63)}.uv_e1wcoel:disabled{background:var(--rwwhsl5k);color:var(--rwwhsl62)}.uv_e1wcoem{background:var(--rwwhsl67);color:var(--rwwhsl6p);border:none}.uv_e1wcoem:hover,.uv_e1wcoem:active{background:var(--rwwhsl69);color:var(--rwwhsl6r)}.uv_e1wcoem:disabled{background:var(--rwwhsl68);color:var(--rwwhsl6q)}.uv_e1wcoen{background:var(--rwwhsl9);color:var(--rwwhslr);border:none}.uv_e1wcoen:hover,.uv_e1wcoen:active{background:var(--rwwhslb);color:var(--rwwhslt)}.uv_e1wcoen:disabled{background:var(--rwwhsla);color:var(--rwwhsls)}.uv_e1wcoeo{background:var(--rwwhslx);color:var(--rwwhsl1f);border:none}.uv_e1wcoeo:hover,.uv_e1wcoeo:active{background:var(--rwwhslz);color:var(--rwwhsl1h)}.uv_e1wcoeo:disabled{background:var(--rwwhsly);color:var(--rwwhsl1g)}.uv_e1wcoep{background:var(--rwwhsl6v);color:var(--rwwhsl7d);border:none}.uv_e1wcoep:hover,.uv_e1wcoep:active{background:var(--rwwhsl6x);color:var(--rwwhsl7f)}.uv_e1wcoep:disabled{background:var(--rwwhsl6w);color:var(--rwwhsl7e)}.uv_e1wcoeq{background:var(--rwwhsl7j);color:var(--rwwhsl81);border:none}.uv_e1wcoeq:hover,.uv_e1wcoeq:active{background:var(--rwwhsl7l);color:var(--rwwhsl83)}.uv_e1wcoeq:disabled{background:var(--rwwhsl7k);color:var(--rwwhsl82)}.uv_e1wcoer{background:var(--rwwhsl1l);color:var(--rwwhsl2m);border:none}.uv_e1wcoer:hover,.uv_e1wcoer:active{background:var(--rwwhsl1q);color:var(--rwwhsl2r)}.uv_e1wcoer:disabled{background:var(--rwwhsl1m);color:var(--rwwhsl2n)}.uv_e1wcoes{background:var(--rwwhsl4r);color:var(--rwwhsl5c);border:none}.uv_e1wcoes:hover,.uv_e1wcoes:active{background:var(--rwwhsl4t);color:var(--rwwhsl5e)}.uv_e1wcoes:disabled{background:var(--rwwhsl4s);color:var(--rwwhsl51)}.uv_e1wcoet{background:var(--rwwhsl53);color:var(--rwwhsl50);border:none}.uv_e1wcoet:hover,.uv_e1wcoet:active{background:var(--rwwhsl55);color:var(--rwwhsl52)}.uv_e1wcoet:disabled{background:var(--rwwhsl54);color:var(--rwwhsl5d)}.uv_e1wcoeu{background:none;color:var(--rwwhsl5y);border:1px solid var(--rwwhsl5m)}.uv_e1wcoeu:hover,.uv_e1wcoeu:active{background:var(--rwwhsl5i);color:var(--rwwhsl60);border:1px solid var(--rwwhsl5o)}.uv_e1wcoeu:disabled{color:var(--rwwhsl5z);border:1px solid var(--rwwhsl5n)}.uv_e1wcoev{background:none;color:var(--rwwhsl6m);border:1px solid var(--rwwhsl6a)}.uv_e1wcoev:hover,.uv_e1wcoev:active{background:var(--rwwhsl66);color:var(--rwwhsl6o);border:1px solid var(--rwwhsl6c)}.uv_e1wcoev:disabled{color:var(--rwwhsl6n);border:1px solid var(--rwwhsl6b)}.uv_e1wcoew{background:none;color:var(--rwwhslo);border:1px solid var(--rwwhslc)}.uv_e1wcoew:hover,.uv_e1wcoew:active{background:var(--rwwhsl8);color:var(--rwwhslq);border:1px solid var(--rwwhsle)}.uv_e1wcoew:disabled{color:var(--rwwhslp);border:1px solid var(--rwwhsld)}.uv_e1wcoex{background:none;color:var(--rwwhsl1c);border:1px solid var(--rwwhsl10)}.uv_e1wcoex:hover,.uv_e1wcoex:active{background:var(--rwwhslw);color:var(--rwwhsl1e);border:1px solid var(--rwwhsl12)}.uv_e1wcoex:disabled{color:var(--rwwhsl1d);border:1px solid var(--rwwhsl11)}.uv_e1wcoey{background:none;color:var(--rwwhsl7a);border:1px solid var(--rwwhsl6y)}.uv_e1wcoey:hover,.uv_e1wcoey:active{background:var(--rwwhsl6u);color:var(--rwwhsl7c);border:1px solid var(--rwwhsl70)}.uv_e1wcoey:disabled{color:var(--rwwhsl7b);border:1px solid var(--rwwhsl6z)}.uv_e1wcoez{background:none;color:var(--rwwhsl7y);border:1px solid var(--rwwhsl7m)}.uv_e1wcoez:hover,.uv_e1wcoez:active{background:var(--rwwhsl7i);color:var(--rwwhsl80);border:1px solid var(--rwwhsl7o)}.uv_e1wcoez:disabled{color:var(--rwwhsl7z);border:1px solid var(--rwwhsl7n)}.uv_e1wcoe10{background:none;color:var(--rwwhsl2j);border:1px solid var(--rwwhsl1y)}.uv_e1wcoe10:hover,.uv_e1wcoe10:active{background:var(--rwwhsl1k);color:var(--rwwhsl2l);border:1px solid var(--rwwhsl23)}.uv_e1wcoe10:disabled{color:var(--rwwhsl2k);border:1px solid var(--rwwhsl1z)}.uv_e1wcoe11{background:none;color:var(--rwwhsl50);border:1px solid var(--rwwhsl4u)}.uv_e1wcoe11:hover,.uv_e1wcoe11:active{background:var(--rwwhsl4t);color:var(--rwwhsl5e);border:1px solid var(--rwwhsl4w)}.uv_e1wcoe11:disabled{color:var(--rwwhsl51);border:1px solid var(--rwwhsl4v)}.uv_e1wcoe12{background:none;color:var(--rwwhsl5c);border:1px solid var(--rwwhsl56)}.uv_e1wcoe12:hover,.uv_e1wcoe12:active{background:var(--rwwhsl55);color:var(--rwwhsl52);border:1px solid var(--rwwhsl58)}.uv_e1wcoe12:disabled{color:var(--rwwhsl5d);border:1px solid var(--rwwhsl57)}.uv_e1wcoe13{background:none;color:var(--rwwhsl5y);border:none}.uv_e1wcoe13:hover,.uv_e1wcoe13:active{background:var(--rwwhsl5i);color:var(--rwwhsl60)}.uv_e1wcoe13:disabled{color:var(--rwwhsl5z)}.uv_e1wcoe14{background:none;color:var(--rwwhsl6m);border:none}.uv_e1wcoe14:hover,.uv_e1wcoe14:active{background:var(--rwwhsl66);color:var(--rwwhsl6o)}.uv_e1wcoe14:disabled{color:var(--rwwhsl6n)}.uv_e1wcoe15{background:none;color:var(--rwwhslo);border:none}.uv_e1wcoe15:hover,.uv_e1wcoe15:active{background:var(--rwwhsl8);color:var(--rwwhslq)}.uv_e1wcoe15:disabled{color:var(--rwwhslp)}.uv_e1wcoe16{background:none;color:var(--rwwhsl1c);border:none}.uv_e1wcoe16:hover,.uv_e1wcoe16:active{background:var(--rwwhslw);color:var(--rwwhsl1e)}.uv_e1wcoe16:disabled{color:var(--rwwhsl1d)}.uv_e1wcoe17{background:none;color:var(--rwwhsl7a);border:none}.uv_e1wcoe17:hover,.uv_e1wcoe17:active{background:var(--rwwhsl6u);color:var(--rwwhsl7c)}.uv_e1wcoe17:disabled{color:var(--rwwhsl7b)}.uv_e1wcoe18{background:none;color:var(--rwwhsl7y);border:none}.uv_e1wcoe18:hover,.uv_e1wcoe18:active{background:var(--rwwhsl7i);color:var(--rwwhsl80)}.uv_e1wcoe18:disabled{color:var(--rwwhsl7z)}.uv_e1wcoe19{background:none;color:var(--rwwhsl2j);border:none}.uv_e1wcoe19:hover,.uv_e1wcoe19:active{background:var(--rwwhsl1k);color:var(--rwwhsl2l)}.uv_e1wcoe19:disabled{color:var(--rwwhsl2k)}.uv_e1wcoe1a{background:none;color:var(--rwwhsl50);border:none}.uv_e1wcoe1a:hover,.uv_e1wcoe1a:active{background:var(--rwwhsl4t);color:var(--rwwhsl5e)}.uv_e1wcoe1a:disabled{color:var(--rwwhsl51)}.uv_e1wcoe1b{background:none;color:var(--rwwhsl5c);border:none}.uv_e1wcoe1b:hover,.uv_e1wcoe1b:active{background:var(--rwwhsl55);color:var(--rwwhsl52)}.uv_e1wcoe1b:disabled{color:var(--rwwhsl5d)}.backdrop-drawer{padding:0!important;transition:opacity .1s ease-in-out}
1
+ .uv_e1wcoe0{display:inline-flex;position:relative;flex-direction:row;box-sizing:border-box;align-items:center;justify-content:center;outline-offset:2px;white-space:nowrap;text-decoration:none;border-radius:var(--rwwhsl85)}.uv_e1wcoe0:hover{text-decoration:none}.uv_e1wcoe1{height:var(--rwwhsl9k);padding-left:var(--rwwhsl9v);padding-right:var(--rwwhsl9v);gap:var(--rwwhsl9u);font-size:var(--rwwhslb9);font-family:var(--rwwhslb8);font-weight:var(--rwwhslbg);letter-spacing:var(--rwwhslbb);line-height:var(--rwwhslbc)}.uv_e1wcoe2{height:var(--rwwhsl9j);padding-left:var(--rwwhsla6);padding-right:var(--rwwhsla6);gap:var(--rwwhsl9u);font-size:var(--rwwhslar);font-family:var(--rwwhslaq);font-weight:var(--rwwhslay);letter-spacing:var(--rwwhslat);line-height:var(--rwwhslau)}.uv_e1wcoe3{height:var(--rwwhsl9h);padding-left:var(--rwwhsl9u);padding-right:var(--rwwhsl9u);gap:var(--rwwhsl9u);font-size:var(--rwwhslar);font-family:var(--rwwhslaq);font-weight:var(--rwwhslay);letter-spacing:var(--rwwhslat);line-height:var(--rwwhslau)}.uv_e1wcoe4{height:var(--rwwhsl9f);padding-left:var(--rwwhsla4);padding-right:var(--rwwhsla4);gap:var(--rwwhsla4);font-size:var(--rwwhslar);font-family:var(--rwwhslaq);font-weight:var(--rwwhslay);letter-spacing:var(--rwwhslat);line-height:var(--rwwhslau)}.uv_e1wcoe5{width:100%}.uv_e1wcoe6{width:auto}.uv_e1wcoej{cursor:not-allowed}.uv_e1wcoek{cursor:pointer}.uv_e1wcoel{background:var(--rwwhsl5j);color:var(--rwwhsl61);border:none}.uv_e1wcoel:hover,.uv_e1wcoel:active{background:var(--rwwhsl5l);color:var(--rwwhsl63)}.uv_e1wcoel:disabled{background:var(--rwwhsl5k);color:var(--rwwhsl62)}.uv_e1wcoem{background:var(--rwwhsl67);color:var(--rwwhsl6p);border:none}.uv_e1wcoem:hover,.uv_e1wcoem:active{background:var(--rwwhsl69);color:var(--rwwhsl6r)}.uv_e1wcoem:disabled{background:var(--rwwhsl68);color:var(--rwwhsl6q)}.uv_e1wcoen{background:var(--rwwhsl9);color:var(--rwwhslr);border:none}.uv_e1wcoen:hover,.uv_e1wcoen:active{background:var(--rwwhslb);color:var(--rwwhslt)}.uv_e1wcoen:disabled{background:var(--rwwhsla);color:var(--rwwhsls)}.uv_e1wcoeo{background:var(--rwwhslx);color:var(--rwwhsl1f);border:none}.uv_e1wcoeo:hover,.uv_e1wcoeo:active{background:var(--rwwhslz);color:var(--rwwhsl1h)}.uv_e1wcoeo:disabled{background:var(--rwwhsly);color:var(--rwwhsl1g)}.uv_e1wcoep{background:var(--rwwhsl6v);color:var(--rwwhsl7d);border:none}.uv_e1wcoep:hover,.uv_e1wcoep:active{background:var(--rwwhsl6x);color:var(--rwwhsl7f)}.uv_e1wcoep:disabled{background:var(--rwwhsl6w);color:var(--rwwhsl7e)}.uv_e1wcoeq{background:var(--rwwhsl7j);color:var(--rwwhsl81);border:none}.uv_e1wcoeq:hover,.uv_e1wcoeq:active{background:var(--rwwhsl7l);color:var(--rwwhsl83)}.uv_e1wcoeq:disabled{background:var(--rwwhsl7k);color:var(--rwwhsl82)}.uv_e1wcoer{background:var(--rwwhsl1l);color:var(--rwwhsl2m);border:none}.uv_e1wcoer:hover,.uv_e1wcoer:active{background:var(--rwwhsl1q);color:var(--rwwhsl2r)}.uv_e1wcoer:disabled{background:var(--rwwhsl1m);color:var(--rwwhsl2n)}.uv_e1wcoes{background:var(--rwwhsl4r);color:var(--rwwhsl5c);border:none}.uv_e1wcoes:hover,.uv_e1wcoes:active{background:var(--rwwhsl4t);color:var(--rwwhsl5e)}.uv_e1wcoes:disabled{background:var(--rwwhsl4s);color:var(--rwwhsl51)}.uv_e1wcoet{background:var(--rwwhsl53);color:var(--rwwhsl50);border:none}.uv_e1wcoet:hover,.uv_e1wcoet:active{background:var(--rwwhsl55);color:var(--rwwhsl52)}.uv_e1wcoet:disabled{background:var(--rwwhsl54);color:var(--rwwhsl5d)}.uv_e1wcoeu{background:none;color:var(--rwwhsl5y);border:1px solid var(--rwwhsl5m)}.uv_e1wcoeu:hover,.uv_e1wcoeu:active{background:var(--rwwhsl5i);color:var(--rwwhsl60);border:1px solid var(--rwwhsl5o)}.uv_e1wcoeu:disabled{color:var(--rwwhsl5z);border:1px solid var(--rwwhsl5n)}.uv_e1wcoev{background:none;color:var(--rwwhsl6m);border:1px solid var(--rwwhsl6a)}.uv_e1wcoev:hover,.uv_e1wcoev:active{background:var(--rwwhsl66);color:var(--rwwhsl6o);border:1px solid var(--rwwhsl6c)}.uv_e1wcoev:disabled{color:var(--rwwhsl6n);border:1px solid var(--rwwhsl6b)}.uv_e1wcoew{background:none;color:var(--rwwhslo);border:1px solid var(--rwwhslc)}.uv_e1wcoew:hover,.uv_e1wcoew:active{background:var(--rwwhsl8);color:var(--rwwhslq);border:1px solid var(--rwwhsle)}.uv_e1wcoew:disabled{color:var(--rwwhslp);border:1px solid var(--rwwhsld)}.uv_e1wcoex{background:none;color:var(--rwwhsl1c);border:1px solid var(--rwwhsl10)}.uv_e1wcoex:hover,.uv_e1wcoex:active{background:var(--rwwhslw);color:var(--rwwhsl1e);border:1px solid var(--rwwhsl12)}.uv_e1wcoex:disabled{color:var(--rwwhsl1d);border:1px solid var(--rwwhsl11)}.uv_e1wcoey{background:none;color:var(--rwwhsl7a);border:1px solid var(--rwwhsl6y)}.uv_e1wcoey:hover,.uv_e1wcoey:active{background:var(--rwwhsl6u);color:var(--rwwhsl7c);border:1px solid var(--rwwhsl70)}.uv_e1wcoey:disabled{color:var(--rwwhsl7b);border:1px solid var(--rwwhsl6z)}.uv_e1wcoez{background:none;color:var(--rwwhsl7y);border:1px solid var(--rwwhsl7m)}.uv_e1wcoez:hover,.uv_e1wcoez:active{background:var(--rwwhsl7i);color:var(--rwwhsl80);border:1px solid var(--rwwhsl7o)}.uv_e1wcoez:disabled{color:var(--rwwhsl7z);border:1px solid var(--rwwhsl7n)}.uv_e1wcoe10{background:none;color:var(--rwwhsl2j);border:1px solid var(--rwwhsl1y)}.uv_e1wcoe10:hover,.uv_e1wcoe10:active{background:var(--rwwhsl1k);color:var(--rwwhsl2l);border:1px solid var(--rwwhsl23)}.uv_e1wcoe10:disabled{color:var(--rwwhsl2k);border:1px solid var(--rwwhsl1z)}.uv_e1wcoe11{background:none;color:var(--rwwhsl50);border:1px solid var(--rwwhsl4u)}.uv_e1wcoe11:hover,.uv_e1wcoe11:active{background:var(--rwwhsl4t);color:var(--rwwhsl5e);border:1px solid var(--rwwhsl4w)}.uv_e1wcoe11:disabled{color:var(--rwwhsl51);border:1px solid var(--rwwhsl4v)}.uv_e1wcoe12{background:none;color:var(--rwwhsl5c);border:1px solid var(--rwwhsl56)}.uv_e1wcoe12:hover,.uv_e1wcoe12:active{background:var(--rwwhsl55);color:var(--rwwhsl52);border:1px solid var(--rwwhsl58)}.uv_e1wcoe12:disabled{color:var(--rwwhsl5d);border:1px solid var(--rwwhsl57)}.uv_e1wcoe13{background:none;color:var(--rwwhsl5y);border:none}.uv_e1wcoe13:hover,.uv_e1wcoe13:active{background:var(--rwwhsl5i);color:var(--rwwhsl60)}.uv_e1wcoe13:disabled{color:var(--rwwhsl5z)}.uv_e1wcoe14{background:none;color:var(--rwwhsl6m);border:none}.uv_e1wcoe14:hover,.uv_e1wcoe14:active{background:var(--rwwhsl66);color:var(--rwwhsl6o)}.uv_e1wcoe14:disabled{color:var(--rwwhsl6n)}.uv_e1wcoe15{background:none;color:var(--rwwhslo);border:none}.uv_e1wcoe15:hover,.uv_e1wcoe15:active{background:var(--rwwhsl8);color:var(--rwwhslq)}.uv_e1wcoe15:disabled{color:var(--rwwhslp)}.uv_e1wcoe16{background:none;color:var(--rwwhsl1c);border:none}.uv_e1wcoe16:hover,.uv_e1wcoe16:active{background:var(--rwwhslw);color:var(--rwwhsl1e)}.uv_e1wcoe16:disabled{color:var(--rwwhsl1d)}.uv_e1wcoe17{background:none;color:var(--rwwhsl7a);border:none}.uv_e1wcoe17:hover,.uv_e1wcoe17:active{background:var(--rwwhsl6u);color:var(--rwwhsl7c)}.uv_e1wcoe17:disabled{color:var(--rwwhsl7b)}.uv_e1wcoe18{background:none;color:var(--rwwhsl7y);border:none}.uv_e1wcoe18:hover,.uv_e1wcoe18:active{background:var(--rwwhsl7i);color:var(--rwwhsl80)}.uv_e1wcoe18:disabled{color:var(--rwwhsl7z)}.uv_e1wcoe19{background:none;color:var(--rwwhsl2j);border:none}.uv_e1wcoe19:hover,.uv_e1wcoe19:active{background:var(--rwwhsl1k);color:var(--rwwhsl2l)}.uv_e1wcoe19:disabled{color:var(--rwwhsl2k)}.uv_e1wcoe1a{background:none;color:var(--rwwhsl50);border:none}.uv_e1wcoe1a:hover,.uv_e1wcoe1a:active{background:var(--rwwhsl4t);color:var(--rwwhsl5e)}.uv_e1wcoe1a:disabled{color:var(--rwwhsl51)}.uv_e1wcoe1b{background:none;color:var(--rwwhsl5c);border:none}.uv_e1wcoe1b:hover,.uv_e1wcoe1b:active{background:var(--rwwhsl55);color:var(--rwwhsl52)}.uv_e1wcoe1b:disabled{color:var(--rwwhsl5d)}.backdrop-drawer{padding:0!important;transition:opacity .1s ease-in-out}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "3.0.0-beta.1",
3
+ "version": "3.0.0-beta.3",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {
@@ -89,8 +89,8 @@
89
89
  "deepmerge": "4.3.1",
90
90
  "react-toastify": "11.0.5",
91
91
  "react-use-clipboard": "1.0.9",
92
- "@ultraviolet/icons": "4.0.4-beta.0",
93
- "@ultraviolet/themes": "2.1.0-beta.0"
92
+ "@ultraviolet/themes": "2.1.0-beta.0",
93
+ "@ultraviolet/icons": "4.0.4-beta.0"
94
94
  },
95
95
  "scripts": {
96
96
  "type:generate": "tsc --declaration -p tsconfig.build.json",