@ultraviolet/ui 1.73.2 → 1.74.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/CopyButton/index.cjs +6 -2
- package/dist/components/CopyButton/index.d.ts +2 -1
- package/dist/components/CopyButton/index.js +6 -2
- package/dist/components/SelectInputV2/Dropdown.cjs +10 -10
- package/dist/components/SelectInputV2/Dropdown.js +10 -10
- package/dist/components/Snippet/index.cjs +12 -11
- package/dist/components/Snippet/index.d.ts +2 -1
- package/dist/components/Snippet/index.js +12 -11
- package/dist/components/Stepper/Step.cjs +4 -4
- package/dist/components/Stepper/Step.js +4 -4
- package/dist/components/TextArea/index.cjs +19 -7
- package/dist/components/TextArea/index.d.ts +4 -1
- package/dist/components/TextArea/index.js +20 -8
- package/package.json +2 -2
|
@@ -2,6 +2,7 @@
|
|
|
2
2
|
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
3
3
|
const jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
4
4
|
const _styled = require("@emotion/styled/base");
|
|
5
|
+
const react = require("@emotion/react");
|
|
5
6
|
const icons = require("@ultraviolet/icons");
|
|
6
7
|
const React = require("react");
|
|
7
8
|
const index$1 = require("../Button/index.cjs");
|
|
@@ -26,7 +27,7 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled__default.default("div", pr
|
|
|
26
27
|
} : {
|
|
27
28
|
name: "8k1832",
|
|
28
29
|
styles: "position:relative;display:flex",
|
|
29
|
-
map: "/*# 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":"AAgBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo } from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\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([data-disabled='true']):hover {\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  rows?: 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      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\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows}\n              ref={ref}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
|
|
30
|
+
map: "/*# 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":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable.\n   */\n  rows?: number | 'auto'\n  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      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      if (textArea && rows === 'auto') {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`\n      }\n    }, [value, rows, theme])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : undefined}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
|
|
30
31
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
31
32
|
});
|
|
32
33
|
const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -38,7 +39,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(ind
|
|
|
38
39
|
theme
|
|
39
40
|
}) => theme.space["1.5"], ";right:", ({
|
|
40
41
|
theme
|
|
41
|
-
}) => 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":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo } from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\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([data-disabled='true']):hover {\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  rows?: 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      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\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows}\n              ref={ref}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
42
|
+
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA6BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable.\n   */\n  rows?: number | 'auto'\n  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      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      if (textArea && rows === 'auto') {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`\n      }\n    }, [value, rows, theme])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : undefined}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
42
43
|
const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", process.env.NODE_ENV === "production" ? {
|
|
43
44
|
shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
|
|
44
45
|
target: "enu776d0"
|
|
@@ -73,7 +74,7 @@ const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", proc
|
|
|
73
74
|
theme
|
|
74
75
|
}) => theme.colors.neutral.backgroundWeak, ";border-color:", ({
|
|
75
76
|
theme
|
|
76
|
-
}) => theme.colors.neutral.border, ";}
|
|
77
|
+
}) => theme.colors.neutral.border, ";}&:disabled{background:", ({
|
|
77
78
|
theme
|
|
78
79
|
}) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
|
|
79
80
|
theme
|
|
@@ -81,13 +82,13 @@ const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", proc
|
|
|
81
82
|
theme
|
|
82
83
|
}) => theme.colors.neutral.textDisabled, ";&::placeholder{color:", ({
|
|
83
84
|
theme
|
|
84
|
-
}) => theme.colors.neutral.textWeakDisabled, ";}}&:not(
|
|
85
|
+
}) => theme.colors.neutral.textWeakDisabled, ";}}&:not(:disabled){&:hover{border-color:", ({
|
|
85
86
|
theme
|
|
86
87
|
}) => theme.colors.primary.border, ";}&:focus{outline:none;border-color:", ({
|
|
87
88
|
theme
|
|
88
89
|
}) => theme.colors.primary.border, ";box-shadow:", ({
|
|
89
90
|
theme
|
|
90
|
-
}) => 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":"AAkCuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo } from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\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([data-disabled='true']):hover {\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  rows?: 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      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\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows}\n              ref={ref}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
91
|
+
}) => theme.shadows.focusPrimary, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA0CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable.\n   */\n  rows?: number | 'auto'\n  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      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      if (textArea && rows === 'auto') {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`\n      }\n    }, [value, rows, theme])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : undefined}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
91
92
|
const TextArea = React.forwardRef(({
|
|
92
93
|
id,
|
|
93
94
|
className,
|
|
@@ -117,6 +118,17 @@ const TextArea = React.forwardRef(({
|
|
|
117
118
|
"aria-label": ariaLabel
|
|
118
119
|
}, ref) => {
|
|
119
120
|
const localId = React.useId();
|
|
121
|
+
const theme = react.useTheme();
|
|
122
|
+
const textAreaRef = React.useRef(null);
|
|
123
|
+
React.useImperativeHandle(ref, () => textAreaRef.current);
|
|
124
|
+
React.useEffect(() => {
|
|
125
|
+
const textArea = textAreaRef.current;
|
|
126
|
+
if (textArea && rows === "auto") {
|
|
127
|
+
textArea.style.height = "auto";
|
|
128
|
+
textArea.style.resize = "none";
|
|
129
|
+
textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`;
|
|
130
|
+
}
|
|
131
|
+
}, [value, rows, theme]);
|
|
120
132
|
const sentiment = React.useMemo(() => {
|
|
121
133
|
if (error) {
|
|
122
134
|
return "danger";
|
|
@@ -137,9 +149,9 @@ const TextArea = React.forwardRef(({
|
|
|
137
149
|
labelDescription ?? null
|
|
138
150
|
] }) : null,
|
|
139
151
|
/* @__PURE__ */ jsxRuntime.jsx(index$3.Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsxs(StyledTextAreaWrapper, { children: [
|
|
140
|
-
/* @__PURE__ */ jsxRuntime.jsx(StyledTextArea, { "aria-invalid": !!error, id: id ?? localId, tabIndex, autoFocus, disabled, rows, ref, value, onChange: (event) => {
|
|
152
|
+
/* @__PURE__ */ jsxRuntime.jsx(StyledTextArea, { "aria-invalid": !!error, id: id ?? localId, tabIndex, autoFocus, disabled, rows: rows !== "auto" ? rows : void 0, ref: textAreaRef, value, onChange: (event) => {
|
|
141
153
|
onChange(event.currentTarget.value);
|
|
142
|
-
}, hasSentimentIcon: !!success || !!error, "data-
|
|
154
|
+
}, hasSentimentIcon: !!success || !!error, "data-readonly": readOnly, "data-success": !!success, "data-error": !!error, isClearable: !!computedClearable, minLength, maxLength, placeholder, "data-testid": dataTestId, name, onFocus, onBlur, onKeyDown, "aria-label": ariaLabel }),
|
|
143
155
|
/* @__PURE__ */ jsxRuntime.jsxs(StyledTextAreaAbsoluteStack, { direction: "row", alignItems: "center", gap: "1", children: [
|
|
144
156
|
computedClearable ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Button, { "aria-label": "clear value", variant: "ghost", size: "xsmall", icon: "close", onClick: () => {
|
|
145
157
|
onChange("");
|
|
@@ -37,7 +37,10 @@ type TextAreaProps = {
|
|
|
37
37
|
* Ignored if following props are provided : readyOnly, success.
|
|
38
38
|
*/
|
|
39
39
|
helper?: ReactNode;
|
|
40
|
-
|
|
40
|
+
/**
|
|
41
|
+
* Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable.
|
|
42
|
+
*/
|
|
43
|
+
rows?: number | 'auto';
|
|
41
44
|
minLength?: number;
|
|
42
45
|
maxLength?: number;
|
|
43
46
|
tooltip?: string;
|
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import { jsxs, jsx } from "@emotion/react/jsx-runtime";
|
|
2
2
|
import _styled from "@emotion/styled/base";
|
|
3
|
+
import { useTheme } from "@emotion/react";
|
|
3
4
|
import { AsteriskIcon, CheckCircleIcon, AlertCircleIcon } from "@ultraviolet/icons";
|
|
4
|
-
import { forwardRef, useId, useMemo } from "react";
|
|
5
|
+
import { forwardRef, useId, useRef, useImperativeHandle, useEffect, useMemo } from "react";
|
|
5
6
|
import { SIZE_HEIGHT, Button } from "../Button/index.js";
|
|
6
7
|
import { Row } from "../Row/index.js";
|
|
7
8
|
import { Stack } from "../Stack/index.js";
|
|
@@ -22,7 +23,7 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_EN
|
|
|
22
23
|
} : {
|
|
23
24
|
name: "8k1832",
|
|
24
25
|
styles: "position:relative;display:flex",
|
|
25
|
-
map: "/*# 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":"AAgBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo } from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\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([data-disabled='true']):hover {\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  rows?: 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      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\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows}\n              ref={ref}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
|
|
26
|
+
map: "/*# 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":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable.\n   */\n  rows?: number | 'auto'\n  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      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      if (textArea && rows === 'auto') {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`\n      }\n    }, [value, rows, theme])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : undefined}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
|
|
26
27
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
27
28
|
});
|
|
28
29
|
const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -34,7 +35,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.N
|
|
|
34
35
|
theme
|
|
35
36
|
}) => theme.space["1.5"], ";right:", ({
|
|
36
37
|
theme
|
|
37
|
-
}) => 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":"AAqBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo } from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\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([data-disabled='true']):hover {\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  rows?: 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      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\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows}\n              ref={ref}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
38
|
+
}) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA6BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable.\n   */\n  rows?: number | 'auto'\n  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      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      if (textArea && rows === 'auto') {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`\n      }\n    }, [value, rows, theme])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : undefined}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
38
39
|
const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV === "production" ? {
|
|
39
40
|
shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
|
|
40
41
|
target: "enu776d0"
|
|
@@ -69,7 +70,7 @@ const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV
|
|
|
69
70
|
theme
|
|
70
71
|
}) => theme.colors.neutral.backgroundWeak, ";border-color:", ({
|
|
71
72
|
theme
|
|
72
|
-
}) => theme.colors.neutral.border, ";}
|
|
73
|
+
}) => theme.colors.neutral.border, ";}&:disabled{background:", ({
|
|
73
74
|
theme
|
|
74
75
|
}) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
|
|
75
76
|
theme
|
|
@@ -77,13 +78,13 @@ const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV
|
|
|
77
78
|
theme
|
|
78
79
|
}) => theme.colors.neutral.textDisabled, ";&::placeholder{color:", ({
|
|
79
80
|
theme
|
|
80
|
-
}) => theme.colors.neutral.textWeakDisabled, ";}}&:not(
|
|
81
|
+
}) => theme.colors.neutral.textWeakDisabled, ";}}&:not(:disabled){&:hover{border-color:", ({
|
|
81
82
|
theme
|
|
82
83
|
}) => theme.colors.primary.border, ";}&:focus{outline:none;border-color:", ({
|
|
83
84
|
theme
|
|
84
85
|
}) => theme.colors.primary.border, ";box-shadow:", ({
|
|
85
86
|
theme
|
|
86
|
-
}) => 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":"AAkCuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport { forwardRef, useId, useMemo } from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &[data-disabled='true'] {\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([data-disabled='true']):hover {\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  rows?: 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      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\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows}\n              ref={ref}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-disabled={disabled}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
87
|
+
}) => theme.shadows.focusPrimary, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA0CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable.\n   */\n  rows?: number | 'auto'\n  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      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      if (textArea && rows === 'auto') {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`\n      }\n    }, [value, rows, theme])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : undefined}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
87
88
|
const TextArea = forwardRef(({
|
|
88
89
|
id,
|
|
89
90
|
className,
|
|
@@ -113,6 +114,17 @@ const TextArea = forwardRef(({
|
|
|
113
114
|
"aria-label": ariaLabel
|
|
114
115
|
}, ref) => {
|
|
115
116
|
const localId = useId();
|
|
117
|
+
const theme = useTheme();
|
|
118
|
+
const textAreaRef = useRef(null);
|
|
119
|
+
useImperativeHandle(ref, () => textAreaRef.current);
|
|
120
|
+
useEffect(() => {
|
|
121
|
+
const textArea = textAreaRef.current;
|
|
122
|
+
if (textArea && rows === "auto") {
|
|
123
|
+
textArea.style.height = "auto";
|
|
124
|
+
textArea.style.resize = "none";
|
|
125
|
+
textArea.style.height = `${textArea.scrollHeight === 61 ? 46 : textArea.scrollHeight + 2}px`;
|
|
126
|
+
}
|
|
127
|
+
}, [value, rows, theme]);
|
|
116
128
|
const sentiment = useMemo(() => {
|
|
117
129
|
if (error) {
|
|
118
130
|
return "danger";
|
|
@@ -133,9 +145,9 @@ const TextArea = forwardRef(({
|
|
|
133
145
|
labelDescription ?? null
|
|
134
146
|
] }) : null,
|
|
135
147
|
/* @__PURE__ */ jsx(Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxs(StyledTextAreaWrapper, { children: [
|
|
136
|
-
/* @__PURE__ */ jsx(StyledTextArea, { "aria-invalid": !!error, id: id ?? localId, tabIndex, autoFocus, disabled, rows, ref, value, onChange: (event) => {
|
|
148
|
+
/* @__PURE__ */ jsx(StyledTextArea, { "aria-invalid": !!error, id: id ?? localId, tabIndex, autoFocus, disabled, rows: rows !== "auto" ? rows : void 0, ref: textAreaRef, value, onChange: (event) => {
|
|
137
149
|
onChange(event.currentTarget.value);
|
|
138
|
-
}, hasSentimentIcon: !!success || !!error, "data-
|
|
150
|
+
}, hasSentimentIcon: !!success || !!error, "data-readonly": readOnly, "data-success": !!success, "data-error": !!error, isClearable: !!computedClearable, minLength, maxLength, placeholder, "data-testid": dataTestId, name, onFocus, onBlur, onKeyDown, "aria-label": ariaLabel }),
|
|
139
151
|
/* @__PURE__ */ jsxs(StyledTextAreaAbsoluteStack, { direction: "row", alignItems: "center", gap: "1", children: [
|
|
140
152
|
computedClearable ? /* @__PURE__ */ jsx(Button, { "aria-label": "clear value", variant: "ghost", size: "xsmall", icon: "close", onClick: () => {
|
|
141
153
|
onChange("");
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ultraviolet/ui",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.74.0",
|
|
4
4
|
"description": "Ultraviolet UI",
|
|
5
5
|
"homepage": "https://github.com/scaleway/ultraviolet#readme",
|
|
6
6
|
"repository": {
|
|
@@ -85,7 +85,7 @@
|
|
|
85
85
|
"react-toastify": "10.0.6",
|
|
86
86
|
"react-use-clipboard": "1.0.9",
|
|
87
87
|
"reakit": "1.3.11",
|
|
88
|
-
"@ultraviolet/icons": "3.
|
|
88
|
+
"@ultraviolet/icons": "3.4.0",
|
|
89
89
|
"@ultraviolet/themes": "1.14.2"
|
|
90
90
|
},
|
|
91
91
|
"scripts": {
|