@ultraviolet/ui 1.56.1 → 1.57.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/SearchInput/Key.cjs +30 -0
- package/dist/components/SearchInput/Key.d.ts +6 -0
- package/dist/components/SearchInput/Key.js +28 -0
- package/dist/components/SearchInput/KeyGroup.cjs +10 -0
- package/dist/components/SearchInput/KeyGroup.d.ts +8 -0
- package/dist/components/SearchInput/KeyGroup.js +10 -0
- package/dist/components/SearchInput/index.cjs +38 -7
- package/dist/components/SearchInput/index.d.ts +2 -1
- package/dist/components/SearchInput/index.js +40 -9
- package/dist/components/SearchInput/types.d.ts +5 -1
- package/dist/components/TextArea/index.cjs +4 -4
- package/dist/components/TextArea/index.js +4 -4
- package/dist/components/TextInputV2/index.cjs +9 -6
- package/dist/components/TextInputV2/index.d.ts +40 -0
- package/dist/components/TextInputV2/index.js +9 -6
- package/package.json +5 -5
|
@@ -26,7 +26,7 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled__default.default("div", pr
|
|
|
26
26
|
} : {
|
|
27
27
|
name: "8k1832",
|
|
28
28
|
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":"AAYwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"outlined\" />\n              ) : 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"]} */",
|
|
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":"AAYwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"filled\" />\n              ) : 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
30
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
31
31
|
});
|
|
32
32
|
const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -38,7 +38,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(ind
|
|
|
38
38
|
theme
|
|
39
39
|
}) => theme.space["1.5"], ";right:", ({
|
|
40
40
|
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":"AAiBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"outlined\" />\n              ) : 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"]} */"));
|
|
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":"AAiBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"filled\" />\n              ) : 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
42
|
const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", process.env.NODE_ENV === "production" ? {
|
|
43
43
|
shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
|
|
44
44
|
target: "enu776d0"
|
|
@@ -85,7 +85,7 @@ const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", proc
|
|
|
85
85
|
theme
|
|
86
86
|
}) => theme.colors.primary.border, ";box-shadow:", ({
|
|
87
87
|
theme
|
|
88
|
-
}) => theme.shadows.focusPrimary, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA8BuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"outlined\" />\n              ) : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
|
|
88
|
+
}) => theme.shadows.focusPrimary, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AA8BuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"filled\" />\n              ) : 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"]} */"));
|
|
89
89
|
const TextArea = React.forwardRef(({
|
|
90
90
|
id,
|
|
91
91
|
className,
|
|
@@ -141,7 +141,7 @@ const TextArea = React.forwardRef(({
|
|
|
141
141
|
onChange("");
|
|
142
142
|
}, sentiment: "neutral" }) : null,
|
|
143
143
|
success ? /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "checkbox-circle-outline", color: "success", size: STATE_ICON_SIZE }) : null,
|
|
144
|
-
error ? /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "alert", color: "danger", variant: "
|
|
144
|
+
error ? /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "alert", color: "danger", variant: "filled" }) : null
|
|
145
145
|
] })
|
|
146
146
|
] }) }),
|
|
147
147
|
notice || maxLength ? /* @__PURE__ */ jsxRuntime.jsxs(index$4.Row, { templateColumns: "minmax(0, 1fr) min-content", gap: "1", children: [
|
|
@@ -22,7 +22,7 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled("div", process.env.NODE_EN
|
|
|
22
22
|
} : {
|
|
23
23
|
name: "8k1832",
|
|
24
24
|
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":"AAYwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"outlined\" />\n              ) : 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"]} */",
|
|
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":"AAYwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"filled\" />\n              ) : 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
26
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
27
27
|
});
|
|
28
28
|
const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -34,7 +34,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled(Stack, process.env.N
|
|
|
34
34
|
theme
|
|
35
35
|
}) => theme.space["1.5"], ";right:", ({
|
|
36
36
|
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":"AAiBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"outlined\" />\n              ) : 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"]} */"));
|
|
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":"AAiBiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"filled\" />\n              ) : 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
38
|
const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV === "production" ? {
|
|
39
39
|
shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
|
|
40
40
|
target: "enu776d0"
|
|
@@ -81,7 +81,7 @@ const StyledTextArea = /* @__PURE__ */ _styled("textarea", process.env.NODE_ENV
|
|
|
81
81
|
theme
|
|
82
82
|
}) => theme.colors.primary.border, ";box-shadow:", ({
|
|
83
83
|
theme
|
|
84
|
-
}) => 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":"AA8BuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"outlined\" />\n              ) : 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"]} */"));
|
|
84
|
+
}) => 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":"AA8BuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } 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\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 TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  label: string\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  clearable?: boolean\n  labelDescription?: ReactNode\n}\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      clearable = false,\n      labelDescription,\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                >\n                  {label}\n                </Text>\n                {required ? (\n                  <Icon name=\"asterisk\" color=\"danger\" size={8} />\n                ) : 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            />\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                <Icon\n                  name=\"checkbox-circle-outline\"\n                  color=\"success\"\n                  size={STATE_ICON_SIZE}\n                />\n              ) : null}\n              {error ? (\n                <Icon name=\"alert\" color=\"danger\" variant=\"filled\" />\n              ) : 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"]} */"));
|
|
85
85
|
const TextArea = forwardRef(({
|
|
86
86
|
id,
|
|
87
87
|
className,
|
|
@@ -137,7 +137,7 @@ const TextArea = forwardRef(({
|
|
|
137
137
|
onChange("");
|
|
138
138
|
}, sentiment: "neutral" }) : null,
|
|
139
139
|
success ? /* @__PURE__ */ jsx(Icon, { name: "checkbox-circle-outline", color: "success", size: STATE_ICON_SIZE }) : null,
|
|
140
|
-
error ? /* @__PURE__ */ jsx(Icon, { name: "alert", color: "danger", variant: "
|
|
140
|
+
error ? /* @__PURE__ */ jsx(Icon, { name: "alert", color: "danger", variant: "filled" }) : null
|
|
141
141
|
] })
|
|
142
142
|
] }) }),
|
|
143
143
|
notice || maxLength ? /* @__PURE__ */ jsxs(Row, { templateColumns: "minmax(0, 1fr) min-content", gap: "1", children: [
|