@ultraviolet/ui 1.52.0 → 1.53.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.
@@ -57,7 +57,7 @@ const TagInputContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV ==
57
57
  theme
58
58
  }) => theme.colors.neutral.borderDisabled, ";background:", ({
59
59
  theme
60
- }) => theme.colors.neutral.backgroundDisabled, ";cursor:not-allowed;}" + (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/TagInput/index.tsx"],"names":[],"mappings":"AAqC2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.body.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
60
+ }) => theme.colors.neutral.backgroundDisabled, ";cursor:not-allowed;}" + (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/TagInput/index.tsx"],"names":[],"mappings":"AAqC2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input<{ 'data-size': TagInputSize }>`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                sentiment=\"neutral\"\n                htmlFor={id ?? localId}\n              >\n                {label}\n              </Text>\n              {required ? (\n                <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n              ) : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                  data-size={size}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
61
61
  const DataContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
62
62
  target: "ea7vc6o2"
63
63
  } : {
@@ -65,7 +65,7 @@ const DataContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "p
65
65
  label: "DataContainer"
66
66
  })("height:100%;display:flex;flex-wrap:wrap;align-items:center;gap:", ({
67
67
  theme
68
- }) => theme.space["1"], ";flex: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/TagInput/index.tsx"],"names":[],"mappings":"AA+EmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.body.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
68
+ }) => theme.space["1"], ";flex: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/TagInput/index.tsx"],"names":[],"mappings":"AA+EmC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input<{ 'data-size': TagInputSize }>`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                sentiment=\"neutral\"\n                htmlFor={id ?? localId}\n              >\n                {label}\n              </Text>\n              {required ? (\n                <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n              ) : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                  data-size={size}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
69
69
  const StateContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
70
70
  target: "ea7vc6o1"
71
71
  } : {
@@ -73,7 +73,7 @@ const StateContainer = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "
73
73
  label: "StateContainer"
74
74
  })("display:flex;align-items:center;gap:", ({
75
75
  theme
76
- }) => 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/TagInput/index.tsx"],"names":[],"mappings":"AAwFoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.body.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
76
+ }) => 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/TagInput/index.tsx"],"names":[],"mappings":"AAwFoC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input<{ 'data-size': TagInputSize }>`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                sentiment=\"neutral\"\n                htmlFor={id ?? localId}\n              >\n                {label}\n              </Text>\n              {required ? (\n                <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n              ) : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                  data-size={size}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
77
77
  const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
78
78
  target: "ea7vc6o0"
79
79
  } : {
@@ -81,7 +81,7 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
81
81
  label: "StyledInput"
82
82
  })("display:flex;flex:1;font-size:", ({
83
83
  theme
84
- }) => theme.typography.body.fontSize, ";background:inherit;color:", ({
84
+ }) => theme.typography.bodySmall.fontSize, ";background:inherit;color:", ({
85
85
  theme: {
86
86
  colors
87
87
  }
@@ -89,7 +89,9 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
89
89
  theme: {
90
90
  colors
91
91
  }
92
- }) => colors.neutral.textWeak, ";}height:100%;" + (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/TagInput/index.tsx"],"names":[],"mappings":"AA8FgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.body.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
92
+ }) => colors.neutral.textWeak, ";}height:100%;&[data-size='large']{font-size:", ({
93
+ theme
94
+ }) => theme.typography.body.fontSize, ";}" + (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/TagInput/index.tsx"],"names":[],"mappings":"AA8F+D","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TagInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { getUUID } from '../../utils'\nimport { Button } from '../Button'\nimport { Stack } from '../Stack'\nimport { Tag } from '../Tag'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\n// Size & Padding\nexport const TAGINPUT_SIZE_PADDING = {\n  large: '1.5',\n  medium: '1',\n  small: '0.5',\n} as const\ntype TagInputSize = keyof typeof TAGINPUT_SIZE_PADDING\n\nconst STATUS = {\n  IDLE: 'idle',\n  LOADING: 'loading',\n} as const\n\ntype Keys = keyof typeof STATUS\ntype StatusValue = (typeof STATUS)[Keys]\n\ntype TagInputContainersProps = {\n  size: TagInputSize\n}\nconst TagInputContainer = styled('div', {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<TagInputContainersProps>`\n  display: flex;\n  gap: ${({ theme }) => theme.space['1']};\n  background-color: ${({ theme: { colors } }) => colors.neutral.background};\n\n  padding: ${({ theme, size }) =>\n    `calc(${theme.space[TAGINPUT_SIZE_PADDING[size]]} - 1px) ${theme.space['2']}`};\n  cursor: text;\n\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  border-radius: ${({ theme }) => theme.radii.default};\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\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  &:hover {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-readonly='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n  }\n\n  &[data-disabled='true'] {\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    cursor: not-allowed;\n  }\n`\n\nconst DataContainer = styled('div')`\n  height: 100%;\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n  flex: 1;\n`\n\nconst StateContainer = styled('div')`\n  display: flex;\n  align-items: center;\n  gap: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledInput = styled.input<{ 'data-size': TagInputSize }>`\n  display: flex;\n  flex: 1;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  background: inherit;\n  color: ${({ theme: { colors } }) => colors.neutral.text};\n  border: none;\n  outline: none;\n  &::placeholder {\n    color: ${({ theme: { colors } }) => colors.neutral.textWeak};\n  }\n  height: 100%;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n`\n\nconst convertTagArrayToTagStateArray = (tags?: TagInputProp) =>\n  (tags || [])?.map((tag, index) =>\n    typeof tag === 'object'\n      ? { ...tag, index: getUUID(`tag-${index}`) }\n      : { index: getUUID(`tag-${index}`), label: tag },\n  )\n\ntype TagInputProp = (string | { label: string; index: string })[]\n\ntype TagInputProps = {\n  disabled?: boolean\n  id?: string\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  manualInput?: boolean\n  name?: string\n  onChange?: (tags: string[]) => void\n  /**\n   * @deprecated this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  onChangeError?: (error: Error | string) => void\n  placeholder?: string\n  /**\n   * @deprecated use `value` property instead, both properties work the same way\n   */\n  tags?: TagInputProp\n  value?: TagInputProp\n  /**\n   * @deprecated there is only one variant now, this prop has no more effect\n   */\n  // eslint-disable-next-line react/no-unused-prop-types\n  variant?: string\n  className?: string\n  'data-testid'?: string\n  label?: string\n  /**\n   * Label description displayed right next to the label. It allows you to customize the label content.\n   */\n  labelDescription?: ReactNode\n  required?: boolean\n  size?: TagInputSize\n  error?: string\n  success?: string | boolean\n  helper?: ReactNode\n  readOnly?: boolean\n  tooltip?: string\n  clearable?: boolean\n}\n\n/**\n * TagInput is a component that allows users to input tags.\n */\nexport const TagInput = ({\n  disabled = false,\n  id,\n  name,\n  onChange,\n  placeholder,\n  tags,\n  value,\n  className,\n  'data-testid': dataTestId,\n  label,\n  labelDescription,\n  required = false,\n  size = 'large',\n  error,\n  success,\n  helper,\n  readOnly = false,\n  tooltip,\n  clearable = false,\n}: TagInputProps) => {\n  const tagsProp = value ?? tags\n\n  const [tagInputState, setTagInput] = useState(\n    convertTagArrayToTagStateArray(tagsProp ?? []),\n  )\n  const [input, setInput] = useState<string>('')\n  const [status, setStatus] = useState<{ [key: string]: StatusValue }>({})\n\n  const uniqueId = useId()\n  const localId = id ?? uniqueId\n\n  useEffect(() => {\n    setTagInput(convertTagArrayToTagStateArray(tagsProp))\n  }, [tagsProp, setTagInput])\n\n  const inputRef = useRef<HTMLInputElement>(null)\n\n  const dispatchOnChange = (newState: TagInputProp) => {\n    const changes = newState.map(tag =>\n      typeof tag === 'object' ? tag?.label : tag,\n    )\n\n    onChange?.(changes)\n  }\n\n  const handleContainerClick = () => {\n    if (inputRef.current) {\n      inputRef?.current?.focus()\n    }\n  }\n\n  const onInputChange = (e: ChangeEvent<HTMLInputElement>) =>\n    setInput(e.target.value)\n\n  const addTag = () => {\n    const newTagInput = input\n      ? [...tagInputState, { index: getUUID('tag'), label: input }]\n      : tagInputState\n    setInput('')\n    setTagInput(newTagInput)\n    if (newTagInput.length !== tagInputState.length) {\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const deleteTag = (tagIndex: string) => {\n    setStatus({ [tagIndex]: STATUS.LOADING })\n    const findIndex = tagInputState.findIndex(({ index }) => index === tagIndex)\n    const newTagInput = [...tagInputState]\n    newTagInput.splice(findIndex, 1)\n    try {\n      dispatchOnChange(newTagInput)\n      setTagInput(newTagInput)\n      setStatus({ [tagIndex]: STATUS.IDLE })\n    } catch (e) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const handleInputKeydown: KeyboardEventHandler<HTMLInputElement> = event => {\n    if (event.key === ' ' || event.key === 'Enter') {\n      addTag()\n      event.preventDefault()\n    }\n    if (\n      event.key === 'Backspace' &&\n      inputRef?.current?.selectionStart === 0 &&\n      tagInputState.length\n    ) {\n      event.preventDefault()\n      deleteTag(tagInputState[tagInputState.length - 1].index)\n    }\n  }\n\n  const handlePaste: ClipboardEventHandler<HTMLInputElement> = e => {\n    e.preventDefault()\n    const newTagInput = [\n      ...tagInputState,\n      { index: getUUID('tag'), label: e?.clipboardData?.getData('Text') },\n    ]\n    setTagInput(newTagInput)\n    setStatus({ [newTagInput.length - 1]: STATUS.LOADING })\n    try {\n      dispatchOnChange(newTagInput)\n      setStatus({ [newTagInput.length - 1]: STATUS.IDLE })\n    } catch (err) {\n      setTagInput(tagInputState)\n    }\n  }\n\n  const clearAll = () => {\n    setInput('')\n    setTagInput([])\n    dispatchOnChange([])\n  }\n\n  const helperSentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  const computedClearable = clearable && !!tagInputState.length\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={size === 'large' ? 'bodyStrong' : 'bodySmallStrong'}\n                sentiment=\"neutral\"\n                htmlFor={id ?? localId}\n              >\n                {label}\n              </Text>\n              {required ? (\n                <Icon name=\"asterisk\" sentiment=\"danger\" size={8} />\n              ) : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        <Tooltip text={tooltip}>\n          <TagInputContainer\n            onClick={handleContainerClick}\n            className={className}\n            data-testid={dataTestId}\n            size={size}\n            data-disabled={disabled}\n            data-readonly={readOnly}\n            data-error={!!error}\n            data-success={!!success}\n          >\n            <DataContainer>\n              {tagInputState.map(tag => (\n                <Tag\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                  key={tag.index}\n                  isLoading={status[tag.index] === STATUS.LOADING}\n                  onClose={\n                    !readOnly\n                      ? e => {\n                          e.stopPropagation()\n                          deleteTag(tag.index)\n                        }\n                      : undefined\n                  }\n                >\n                  {tag.label}\n                </Tag>\n              ))}\n              {!disabled ? (\n                <StyledInput\n                  id={localId}\n                  name={name}\n                  aria-label={name}\n                  type=\"text\"\n                  placeholder={!tagInputState.length ? placeholder : ''}\n                  value={input}\n                  onBlur={addTag}\n                  onChange={onInputChange}\n                  onKeyDown={handleInputKeydown}\n                  onPaste={handlePaste}\n                  ref={inputRef}\n                  readOnly={readOnly}\n                  data-size={size}\n                />\n              ) : null}\n            </DataContainer>\n            {computedClearable || success || error ? (\n              <StateContainer>\n                {computedClearable ? (\n                  <Button\n                    aria-label=\"clear value\"\n                    disabled={disabled}\n                    variant=\"ghost\"\n                    size=\"xsmall\"\n                    icon=\"close\"\n                    onClick={clearAll}\n                    sentiment=\"neutral\"\n                  />\n                ) : null}\n                {success ? (\n                  <Icon\n                    name=\"checkbox-circle-outline\"\n                    color=\"success\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n                {error ? (\n                  <Icon\n                    name=\"alert\"\n                    color=\"danger\"\n                    size={16}\n                    disabled={disabled}\n                  />\n                ) : null}\n              </StateContainer>\n            ) : null}\n          </TagInputContainer>\n        </Tooltip>\n      </div>\n      {error || typeof success === 'string' || helper ? (\n        <Text\n          variant=\"caption\"\n          as=\"span\"\n          prominence={!error && !success ? 'weak' : undefined}\n          sentiment={helperSentiment}\n          disabled={disabled || readOnly}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
93
95
  const convertTagArrayToTagStateArray = (tags) => (tags || [])?.map((tag, index) => typeof tag === "object" ? {
94
96
  ...tag,
95
97
  index: getUUID(`tag-${index}`)
@@ -225,8 +227,8 @@ const TagInput = ({
225
227
  return /* @__PURE__ */ jsxs(Stack, { gap: "0.5", className, children: [
226
228
  label || labelDescription ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "1", alignItems: "center", children: [
227
229
  label ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
228
- /* @__PURE__ */ jsx(Text, { as: "label", variant: "bodyStrong", sentiment: "neutral", htmlFor: id ?? localId, children: label }),
229
- required ? /* @__PURE__ */ jsx(Icon, { name: "asterisk", color: "danger", size: 8 }) : null
230
+ /* @__PURE__ */ jsx(Text, { as: "label", variant: size === "large" ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", htmlFor: id ?? localId, children: label }),
231
+ required ? /* @__PURE__ */ jsx(Icon, { name: "asterisk", sentiment: "danger", size: 8 }) : null
230
232
  ] }) : null,
231
233
  labelDescription ?? null
232
234
  ] }) : null,
@@ -236,7 +238,7 @@ const TagInput = ({
236
238
  e.stopPropagation();
237
239
  deleteTag(tag.index);
238
240
  } : void 0, children: tag.label }, tag.index)),
239
- !disabled ? /* @__PURE__ */ jsx(StyledInput, { id: localId, name, "aria-label": name, type: "text", placeholder: !tagInputState.length ? placeholder : "", value: input, onBlur: addTag, onChange: onInputChange, onKeyDown: handleInputKeydown, onPaste: handlePaste, ref: inputRef, readOnly }) : null
241
+ !disabled ? /* @__PURE__ */ jsx(StyledInput, { id: localId, name, "aria-label": name, type: "text", placeholder: !tagInputState.length ? placeholder : "", value: input, onBlur: addTag, onChange: onInputChange, onKeyDown: handleInputKeydown, onPaste: handlePaste, ref: inputRef, readOnly, "data-size": size }) : null
240
242
  ] }),
241
243
  computedClearable || success || error ? /* @__PURE__ */ jsxs(StateContainer, { children: [
242
244
  computedClearable ? /* @__PURE__ */ jsx(Button, { "aria-label": "clear value", disabled, variant: "ghost", size: "xsmall", icon: "close", onClick: clearAll, sentiment: "neutral" }) : null,