@ultraviolet/ui 1.55.3 → 1.56.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) ${\n      theme.space['2']\n    }`};\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({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\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"]} */"));
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) ${\n      theme.space['2']\n    }`};\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 && newTagInput) {\n      setStatus({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      if (newTagInput) {\n        setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n      }\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      if (tagInputState) {\n        deleteTag(tagInputState[tagInputState.length - 1].index)\n      }\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":"AAiFmC","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) ${\n      theme.space['2']\n    }`};\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({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\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"]} */"));
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":"AAiFmC","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) ${\n      theme.space['2']\n    }`};\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 && newTagInput) {\n      setStatus({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      if (newTagInput) {\n        setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n      }\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      if (tagInputState) {\n        deleteTag(tagInputState[tagInputState.length - 1].index)\n      }\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":"AA0FoC","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) ${\n      theme.space['2']\n    }`};\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({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\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"]} */"));
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":"AA0FoC","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) ${\n      theme.space['2']\n    }`};\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 && newTagInput) {\n      setStatus({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      if (newTagInput) {\n        setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n      }\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      if (tagInputState) {\n        deleteTag(tagInputState[tagInputState.length - 1].index)\n      }\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
  } : {
@@ -91,8 +91,8 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
91
91
  }
92
92
  }) => colors.neutral.textWeak, ';}height:100%;&[data-size="large"]{font-size:', ({
93
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":"AAgG+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) ${\n      theme.space['2']\n    }`};\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({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\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"]} */"));
95
- const convertTagArrayToTagStateArray = (tags) => (tags || [])?.map((tag, index) => typeof tag === "object" ? {
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":"AAgG+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) ${\n      theme.space['2']\n    }`};\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 && newTagInput) {\n      setStatus({\n        [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING,\n      })\n    }\n    try {\n      dispatchOnChange(newTagInput)\n      if (newTagInput) {\n        setStatus({ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE })\n      }\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      if (tagInputState) {\n        deleteTag(tagInputState[tagInputState.length - 1].index)\n      }\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"]} */"));
95
+ const convertTagArrayToTagStateArray = (tags) => (tags ?? [])?.map((tag, index) => typeof tag === "object" ? {
96
96
  ...tag,
97
97
  index: getUUID(`tag-${index}`)
98
98
  } : {
@@ -121,7 +121,7 @@ const TagInput = ({
121
121
  clearable = false
122
122
  }) => {
123
123
  const tagsProp = value ?? tags;
124
- const [tagInputState, setTagInput] = useState(convertTagArrayToTagStateArray(tagsProp ?? []));
124
+ const [tagInputState, setTagInput] = useState(convertTagArrayToTagStateArray(tagsProp));
125
125
  const [input, setInput] = useState("");
126
126
  const [status, setStatus] = useState({});
127
127
  const uniqueId = useId();
@@ -147,16 +147,18 @@ const TagInput = ({
147
147
  }] : tagInputState;
148
148
  setInput("");
149
149
  setTagInput(newTagInput);
150
- if (newTagInput.length !== tagInputState.length) {
150
+ if (newTagInput.length !== tagInputState.length && newTagInput) {
151
151
  setStatus({
152
152
  [newTagInput[newTagInput.length - 1].index]: STATUS.LOADING
153
153
  });
154
154
  }
155
155
  try {
156
156
  dispatchOnChange(newTagInput);
157
- setStatus({
158
- [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE
159
- });
157
+ if (newTagInput) {
158
+ setStatus({
159
+ [newTagInput[newTagInput.length - 1].index]: STATUS.IDLE
160
+ });
161
+ }
160
162
  } catch (e) {
161
163
  setTagInput(tagInputState);
162
164
  }
@@ -187,7 +189,9 @@ const TagInput = ({
187
189
  }
188
190
  if (event.key === "Backspace" && inputRef?.current?.selectionStart === 0 && tagInputState.length) {
189
191
  event.preventDefault();
190
- deleteTag(tagInputState[tagInputState.length - 1].index);
192
+ if (tagInputState) {
193
+ deleteTag(tagInputState[tagInputState.length - 1].index);
194
+ }
191
195
  }
192
196
  };
193
197
  const handlePaste = (e) => {
@@ -78,7 +78,7 @@ const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.en
78
78
  &:hover {
79
79
  border: ${colors.neutral.borderDisabled}
80
80
  }
81
- `, ";" + (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/VerificationCode/index.tsx"],"names":[],"mappings":"AA+BE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n} from 'react'\nimport { createRef, useId, useState } from 'react'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: 64,\n  large: 48,\n  medium: 40,\n  small: 32,\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: 56,\n  large: 40,\n  medium: 32,\n  small: 24,\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: solid 1px\n    ${({ 'aria-invalid': error, theme }) =>\n      error ? theme.colors.danger.border : theme.colors.neutral.border};\n  color: ${({ 'aria-invalid': error, theme }) =>\n    error ? theme.colors.danger.text : theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize }) => SIZE_WIDTH[inputSize]}px;\n  height: ${({ inputSize }) => SIZE_HEIGHT[inputSize]}px;\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  &:hover,\n  &:focus {\n    border-color: ${({ 'aria-invalid': error, theme }) =>\n      error\n        ? theme.colors.danger.borderHover\n        : theme.colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ 'aria-invalid': error, theme: { shadows } }) =>\n      error ? shadows.focusDanger : shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};\n\n    &:hover {\n      border: ${colors.neutral.borderDisabled}\n    }\n    `}\n`\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n}\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(\n    new Array(fields).fill(''),\n    initialValue.substring(0, fields).split(''),\n  )\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace':\n          event.preventDefault()\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        case 'ArrowLeft':\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        case 'ArrowRight':\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        case 'ArrowUp':\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        case 'ArrowDown':\n          event.preventDefault()\n          last?.current?.focus()\n          break\n        default:\n          break\n      }\n    }\n\n  const inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n    event.target.select()\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [\n        ...event.clipboardData.getData('Text').split(''),\n      ].map((copiedValue: string) =>\n        // Replace non number char with empty char when type is number\n        type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = vals.slice()\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  return (\n    <div className={className} data-testid={dataTestId}>\n      {values.map((value: string, index: number) => (\n        <StyledInput\n          css={[inputStyle]}\n          aria-invalid={error}\n          inputSize={size}\n          type={type === 'number' ? 'tel' : type}\n          pattern={type === 'number' ? '[0-9]*' : undefined}\n          // eslint-disable-next-line react/no-array-index-key\n          key={index}\n          data-testid={index}\n          value={value}\n          id={`${inputId || uniqueId}-${index}`}\n          ref={inputRefs[index]}\n          onChange={inputOnChange(index)}\n          onKeyDown={inputOnKeyDown(index)}\n          onPaste={inputOnPaste(index)}\n          onFocus={inputOnFocus}\n          disabled={disabled}\n          required={required}\n          placeholder={placeholder?.[index] ?? ''}\n          aria-label={`${ariaLabel} ${index}`}\n        />\n      ))}\n    </div>\n  )\n}\n"]} */"));
81
+ `, ";" + (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/VerificationCode/index.tsx"],"names":[],"mappings":"AA+BE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n} from 'react'\nimport { createRef, useId, useState } from 'react'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: 64,\n  large: 48,\n  medium: 40,\n  small: 32,\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: 56,\n  large: 40,\n  medium: 32,\n  small: 24,\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: solid 1px\n    ${({ 'aria-invalid': error, theme }) =>\n      error ? theme.colors.danger.border : theme.colors.neutral.border};\n  color: ${({ 'aria-invalid': error, theme }) =>\n    error ? theme.colors.danger.text : theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize }) => SIZE_WIDTH[inputSize]}px;\n  height: ${({ inputSize }) => SIZE_HEIGHT[inputSize]}px;\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  &:hover,\n  &:focus {\n    border-color: ${({ 'aria-invalid': error, theme }) =>\n      error\n        ? theme.colors.danger.borderHover\n        : theme.colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ 'aria-invalid': error, theme: { shadows } }) =>\n      error ? shadows.focusDanger : shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};\n\n    &:hover {\n      border: ${colors.neutral.borderDisabled}\n    }\n    `}\n`\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n}\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(\n    new Array(fields).fill(''),\n    initialValue.substring(0, fields).split(''),\n  )\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace':\n          event.preventDefault()\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        case 'ArrowLeft':\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        case 'ArrowRight':\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        case 'ArrowUp':\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        case 'ArrowDown':\n          event.preventDefault()\n          last?.current?.focus()\n          break\n        default:\n          break\n      }\n    }\n\n  const inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n    event.target.select()\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [\n        ...event.clipboardData.getData('Text').split(''),\n      ].map((copiedValue: string) =>\n        // Replace non number char with empty char when type is number\n        type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = vals.slice()\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  return (\n    <div className={className} data-testid={dataTestId}>\n      {values.map((value: string, index: number) => (\n        <StyledInput\n          css={[inputStyle]}\n          aria-invalid={error}\n          inputSize={size}\n          type={type === 'number' ? 'tel' : type}\n          pattern={type === 'number' ? '[0-9]*' : undefined}\n          // eslint-disable-next-line react/no-array-index-key\n          key={index}\n          data-testid={index}\n          value={value}\n          id={`${inputId || uniqueId}-${index}`}\n          ref={inputRefs[index]}\n          onChange={inputOnChange(index)}\n          onKeyDown={inputOnKeyDown(index)}\n          onPaste={inputOnPaste(index)}\n          onFocus={inputOnFocus}\n          disabled={disabled}\n          required={required}\n          placeholder={placeholder?.[index] ?? ''}\n          aria-label={`${ariaLabel} ${index}`}\n        />\n      ))}\n    </div>\n  )\n}\n"]} */"));
82
82
  const DEFAULT_ON_FUNCTION = () => {
83
83
  };
84
84
  const VerificationCode = ({
@@ -127,7 +127,7 @@ const VerificationCode = ({
127
127
  return;
128
128
  }
129
129
  const sanitizedValue = value[0];
130
- newValues[index] = sanitizedValue;
130
+ newValues[index] = sanitizedValue ?? "";
131
131
  setValues(newValues);
132
132
  const nextIndex = Math.min(index + 1, fields - 1);
133
133
  const next = inputRefs[nextIndex];
@@ -195,7 +195,7 @@ const VerificationCode = ({
195
195
  return /* @__PURE__ */ jsxRuntime.jsx("div", { className, "data-testid": dataTestId, children: values.map((value, index) => /* @__PURE__ */ jsxRuntime.jsx(
196
196
  StyledInput,
197
197
  {
198
- css: [inputStyle, process.env.NODE_ENV === "production" ? "" : ";label:VerificationCode;", 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/VerificationCode/index.tsx"],"names":[],"mappings":"AAgSU","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n} from 'react'\nimport { createRef, useId, useState } from 'react'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: 64,\n  large: 48,\n  medium: 40,\n  small: 32,\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: 56,\n  large: 40,\n  medium: 32,\n  small: 24,\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: solid 1px\n    ${({ 'aria-invalid': error, theme }) =>\n      error ? theme.colors.danger.border : theme.colors.neutral.border};\n  color: ${({ 'aria-invalid': error, theme }) =>\n    error ? theme.colors.danger.text : theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize }) => SIZE_WIDTH[inputSize]}px;\n  height: ${({ inputSize }) => SIZE_HEIGHT[inputSize]}px;\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  &:hover,\n  &:focus {\n    border-color: ${({ 'aria-invalid': error, theme }) =>\n      error\n        ? theme.colors.danger.borderHover\n        : theme.colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ 'aria-invalid': error, theme: { shadows } }) =>\n      error ? shadows.focusDanger : shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};\n\n    &:hover {\n      border: ${colors.neutral.borderDisabled}\n    }\n    `}\n`\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n}\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(\n    new Array(fields).fill(''),\n    initialValue.substring(0, fields).split(''),\n  )\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace':\n          event.preventDefault()\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        case 'ArrowLeft':\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        case 'ArrowRight':\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        case 'ArrowUp':\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        case 'ArrowDown':\n          event.preventDefault()\n          last?.current?.focus()\n          break\n        default:\n          break\n      }\n    }\n\n  const inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n    event.target.select()\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [\n        ...event.clipboardData.getData('Text').split(''),\n      ].map((copiedValue: string) =>\n        // Replace non number char with empty char when type is number\n        type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = vals.slice()\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  return (\n    <div className={className} data-testid={dataTestId}>\n      {values.map((value: string, index: number) => (\n        <StyledInput\n          css={[inputStyle]}\n          aria-invalid={error}\n          inputSize={size}\n          type={type === 'number' ? 'tel' : type}\n          pattern={type === 'number' ? '[0-9]*' : undefined}\n          // eslint-disable-next-line react/no-array-index-key\n          key={index}\n          data-testid={index}\n          value={value}\n          id={`${inputId || uniqueId}-${index}`}\n          ref={inputRefs[index]}\n          onChange={inputOnChange(index)}\n          onKeyDown={inputOnKeyDown(index)}\n          onPaste={inputOnPaste(index)}\n          onFocus={inputOnFocus}\n          disabled={disabled}\n          required={required}\n          placeholder={placeholder?.[index] ?? ''}\n          aria-label={`${ariaLabel} ${index}`}\n        />\n      ))}\n    </div>\n  )\n}\n"]} */"],
198
+ css: [inputStyle, process.env.NODE_ENV === "production" ? "" : ";label:VerificationCode;", 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/VerificationCode/index.tsx"],"names":[],"mappings":"AAgSU","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n} from 'react'\nimport { createRef, useId, useState } from 'react'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: 64,\n  large: 48,\n  medium: 40,\n  small: 32,\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: 56,\n  large: 40,\n  medium: 32,\n  small: 24,\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: solid 1px\n    ${({ 'aria-invalid': error, theme }) =>\n      error ? theme.colors.danger.border : theme.colors.neutral.border};\n  color: ${({ 'aria-invalid': error, theme }) =>\n    error ? theme.colors.danger.text : theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize }) => SIZE_WIDTH[inputSize]}px;\n  height: ${({ inputSize }) => SIZE_HEIGHT[inputSize]}px;\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  &:hover,\n  &:focus {\n    border-color: ${({ 'aria-invalid': error, theme }) =>\n      error\n        ? theme.colors.danger.borderHover\n        : theme.colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ 'aria-invalid': error, theme: { shadows } }) =>\n      error ? shadows.focusDanger : shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  ${({ disabled, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};\n\n    &:hover {\n      border: ${colors.neutral.borderDisabled}\n    }\n    `}\n`\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n}\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(\n    new Array(fields).fill(''),\n    initialValue.substring(0, fields).split(''),\n  )\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace':\n          event.preventDefault()\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        case 'ArrowLeft':\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        case 'ArrowRight':\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        case 'ArrowUp':\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        case 'ArrowDown':\n          event.preventDefault()\n          last?.current?.focus()\n          break\n        default:\n          break\n      }\n    }\n\n  const inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n    event.target.select()\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [\n        ...event.clipboardData.getData('Text').split(''),\n      ].map((copiedValue: string) =>\n        // Replace non number char with empty char when type is number\n        type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = vals.slice()\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  return (\n    <div className={className} data-testid={dataTestId}>\n      {values.map((value: string, index: number) => (\n        <StyledInput\n          css={[inputStyle]}\n          aria-invalid={error}\n          inputSize={size}\n          type={type === 'number' ? 'tel' : type}\n          pattern={type === 'number' ? '[0-9]*' : undefined}\n          // eslint-disable-next-line react/no-array-index-key\n          key={index}\n          data-testid={index}\n          value={value}\n          id={`${inputId || uniqueId}-${index}`}\n          ref={inputRefs[index]}\n          onChange={inputOnChange(index)}\n          onKeyDown={inputOnKeyDown(index)}\n          onPaste={inputOnPaste(index)}\n          onFocus={inputOnFocus}\n          disabled={disabled}\n          required={required}\n          placeholder={placeholder?.[index] ?? ''}\n          aria-label={`${ariaLabel} ${index}`}\n        />\n      ))}\n    </div>\n  )\n}\n"]} */"],
199
199
  "aria-invalid": error,
200
200
  inputSize: size,
201
201
  type: type === "number" ? "tel" : type,