@ultraviolet/ui 1.84.4 → 1.85.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.
@@ -17,13 +17,13 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
17
17
  theme
18
18
  }) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
19
19
  theme
20
- }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[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/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { SearchIcon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 38.125rem;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef<HTMLInputElement, SearchInputProps>(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={<SearchIcon disabled={disabled} sentiment=\"neutral\" />}\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={event => onSearchCallback(event.target.value)}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
20
+ }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[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/SearchInput/index.tsx"],"names":[],"mappings":"AAwBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { SearchIcon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 38.125rem;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef<HTMLInputElement, SearchInputProps>(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      defaultValue = '',\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n      value,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState(defaultValue)\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    useEffect(() => {\n      if (value) {\n        setSearchTerms(value)\n      }\n    }, [value])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={<SearchIcon disabled={disabled} sentiment=\"neutral\" />}\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={event => onSearchCallback(event.target.value)}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
21
21
  const StyledTextInputV2 = /* @__PURE__ */ _styled(TextInputV2, process.env.NODE_ENV === "production" ? {
22
22
  target: "eefux4u0"
23
23
  } : {
24
24
  target: "eefux4u0",
25
25
  label: "StyledTextInputV2"
26
- })(BasicPrefixStack, "{border:none;}", StyledInput, "{padding:0;}", BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { SearchIcon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 38.125rem;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef<HTMLInputElement, SearchInputProps>(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState('')\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={<SearchIcon disabled={disabled} sentiment=\"neutral\" />}\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={event => onSearchCallback(event.target.value)}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
26
+ })(BasicPrefixStack, "{border:none;}", StyledInput, "{padding:0;}", BasicSuffixStack, "{border:none;}" + (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/SearchInput/index.tsx"],"names":[],"mappings":"AAiC6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { SearchIcon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useMemo,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\nimport { isClientSide } from '../../helpers/isClientSide'\nimport { Popup } from '../Popup'\nimport {\n  BasicPrefixStack,\n  BasicSuffixStack,\n  StyledInput,\n  TextInputV2,\n} from '../TextInputV2'\nimport { KeyGroup } from './KeyGroup'\nimport type { SearchInputProps } from './types'\n\nconst StyledPopup = styled(Popup)`\n  width: 100%;\n  text-align: initial;\n  min-width: 38.125rem;\n  padding: ${({ theme }) => `${theme.space['2']} ${theme.space['1']}`};\n  background: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n`\n\nconst StyledTextInputV2 = styled(TextInputV2)`\n  ${BasicPrefixStack} {\n    border: none;\n  }\n\n  ${StyledInput} {\n    padding: 0;\n  }\n\n  ${BasicSuffixStack} {\n    border: none;\n  }\n`\n\n/**\n * SearchInput is a component that allows users to search for items. It is a combination of a TextInputV2 and a Popup. The Popup is used to display search results.\n * Children of the SearchInput component can be a function that receives an object with the following properties:\n * - `searchTerms`: the current search terms\n * - `isOpen`: a boolean indicating if the popup is open\n * - `toggleIsOpen`: a function to toggle the popup\n */\nexport const SearchInput = forwardRef<HTMLInputElement, SearchInputProps>(\n  (\n    {\n      placeholder,\n      label,\n      labelDescription,\n      loading,\n      size,\n      popupPlacement,\n      threshold = 0,\n      children,\n      onSearch,\n      onClose,\n      'data-testid': dataTestId,\n      shortcut = false,\n      error,\n      defaultValue = '',\n      disabled,\n      className,\n      minLength,\n      maxLength,\n      tooltip,\n      onFocus,\n      onBlur,\n      name,\n      id,\n      'aria-live': ariaLive,\n      'aria-atomic': ariaAtomic,\n      'aria-labelledby': ariaLabelledby,\n      readOnly,\n      required,\n      autoFocus,\n      autoComplete,\n      onKeyDown,\n      role,\n      value,\n    }: SearchInputProps,\n    ref: Ref<HTMLInputElement>,\n  ) => {\n    const focusedLinkIndex = useRef(0)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const [containerWidth, setContainerWidth] = useState(0)\n    const [searchTerms, setSearchTerms] = useState(defaultValue)\n    const [isMacOS, setIsMacOS] = useState(false)\n    const [keyPressed, setKeyPressed] = useState<string[]>([])\n    const [isOpen, toggleIsOpen] = useReducer(state => !state, false)\n    const innerSearchInputRef = useRef<HTMLInputElement>(null)\n    useImperativeHandle(\n      ref,\n      () => innerSearchInputRef.current as HTMLInputElement,\n    )\n\n    const content =\n      typeof children === 'function'\n        ? children({ searchTerms, isOpen, toggleIsOpen })\n        : children\n\n    const resizeSearchBar = () => {\n      if (popupRef.current) {\n        setContainerWidth(popupRef.current.getBoundingClientRect().width)\n      }\n    }\n\n    const handleNavigation = (event: KeyboardEvent) => {\n      const links = [...(popupRef.current?.querySelectorAll('a') ?? [])]\n\n      if (\n        links.length > 0 &&\n        focusedLinkIndex.current >= 0 &&\n        focusedLinkIndex.current <= links.length\n      ) {\n        if (event.key === 'ArrowUp') {\n          if (focusedLinkIndex.current - 1 < 0) {\n            focusedLinkIndex.current = links.length - 1\n          } else {\n            focusedLinkIndex.current -= 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n\n        if (event.key === 'ArrowDown') {\n          if (focusedLinkIndex.current + 1 >= links.length) {\n            focusedLinkIndex.current = 0\n          } else {\n            focusedLinkIndex.current += 1\n          }\n          links[focusedLinkIndex.current]?.focus()\n        }\n      }\n    }\n\n    useEffect(() => {\n      document.addEventListener('keyup', handleNavigation)\n\n      return () => document.removeEventListener('keyup', handleNavigation)\n    }, [])\n\n    useEffect(() => {\n      resizeSearchBar()\n\n      window.addEventListener('resize', resizeSearchBar)\n\n      return () => window.removeEventListener('resize', resizeSearchBar)\n    }, [])\n\n    useEffect(() => {\n      if (value) {\n        setSearchTerms(value)\n      }\n    }, [value])\n\n    const onSearchCallback = (localValue: string) => {\n      setSearchTerms(localValue)\n\n      try {\n        onSearch(localValue)\n        if (localValue.length >= threshold && !isOpen) {\n          toggleIsOpen()\n        }\n      } catch {\n        toggleIsOpen()\n      }\n    }\n\n    const onCloseCallback = () => {\n      onClose?.()\n      if (isOpen) {\n        toggleIsOpen()\n      }\n    }\n\n    useEffect(() => {\n      if (isClientSide) {\n        // We need to check if window is defined to avoid SSR issues\n        setIsMacOS(navigator.userAgent.includes('Mac'))\n      }\n    }, [])\n\n    const handleKeyPressed = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { ctrlKey, metaKey, key } = event\n        setKeyPressed([...keyPressed, key.toUpperCase()])\n\n        if (typeof shortcut === 'boolean') {\n          if (\n            (key === 'k' || key === 'K') &&\n            ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        } else {\n          const uppercaseShortcut = shortcut.map(s => s.toUpperCase())\n\n          if (\n            JSON.stringify([...keyPressed, key.toUpperCase()]) ===\n            JSON.stringify(uppercaseShortcut)\n          ) {\n            event.preventDefault()\n            innerSearchInputRef.current?.focus()\n          }\n        }\n      },\n      [keyPressed, shortcut, isMacOS],\n    )\n\n    const handleKeyReleased = useCallback(\n      (event: KeyboardEvent) => {\n        if (!(event instanceof KeyboardEvent)) {\n          return\n        }\n\n        const { key } = event\n        setKeyPressed(keyPressed.filter(k => k !== key.toUpperCase()))\n      },\n      [keyPressed],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleKeyPressed)\n        document.body.addEventListener('keyup', handleKeyReleased)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleKeyPressed)\n        document.body.removeEventListener('keyup', handleKeyReleased)\n      }\n    }, [shortcut, disabled, handleKeyPressed, handleKeyReleased])\n\n    const keys = useMemo(() => {\n      if (typeof shortcut === 'boolean') {\n        return [isMacOS ? '⌘' : 'Ctrl', 'K']\n      }\n\n      const filteredKey = shortcut.map(key => {\n        if (key === 'Meta') {\n          return '⌘'\n        }\n\n        if (key === 'Control') {\n          return 'Ctrl'\n        }\n\n        return key\n      })\n\n      return filteredKey\n    }, [isMacOS, shortcut])\n\n    return (\n      <div style={{ width: '100%' }}>\n        <StyledPopup\n          data-testid={`popup-${dataTestId}`}\n          role=\"dialog\"\n          visible={isOpen}\n          onClose={onCloseCallback}\n          placement={popupPlacement}\n          maxWidth={containerWidth}\n          hideOnClickOutside\n          hasArrow={false}\n          innerRef={popupRef}\n          text={content}\n          maxHeight={410}\n          debounceDelay={0}\n        >\n          <StyledTextInputV2\n            ref={innerSearchInputRef}\n            prefix={<SearchIcon disabled={disabled} sentiment=\"neutral\" />}\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup disabled={disabled} keys={keys} />\n              ) : undefined\n            }\n            data-testid={dataTestId}\n            error={error}\n            value={searchTerms}\n            size={size}\n            label={label}\n            placeholder={placeholder}\n            loading={loading}\n            onChange={event => onSearchCallback(event.target.value)}\n            clearable\n            disabled={disabled}\n            className={className}\n            aria-atomic={ariaAtomic}\n            autoComplete={autoComplete}\n            aria-labelledby={ariaLabelledby}\n            aria-live={ariaLive}\n            id={id}\n            name={name}\n            readOnly={readOnly}\n            required={required}\n            autoFocus={autoFocus}\n            maxLength={maxLength}\n            minLength={minLength}\n            onBlur={onBlur}\n            onFocus={onFocus}\n            onKeyDown={onKeyDown}\n            role={role}\n            tooltip={tooltip}\n            labelDescription={labelDescription}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
27
27
  const SearchInput = forwardRef(({
28
28
  placeholder,
29
29
  label,
@@ -38,6 +38,7 @@ const SearchInput = forwardRef(({
38
38
  "data-testid": dataTestId,
39
39
  shortcut = false,
40
40
  error,
41
+ defaultValue = "",
41
42
  disabled,
42
43
  className,
43
44
  minLength,
@@ -55,12 +56,13 @@ const SearchInput = forwardRef(({
55
56
  autoFocus,
56
57
  autoComplete,
57
58
  onKeyDown,
58
- role
59
+ role,
60
+ value
59
61
  }, ref) => {
60
62
  const focusedLinkIndex = useRef(0);
61
63
  const popupRef = useRef(null);
62
64
  const [containerWidth, setContainerWidth] = useState(0);
63
- const [searchTerms, setSearchTerms] = useState("");
65
+ const [searchTerms, setSearchTerms] = useState(defaultValue);
64
66
  const [isMacOS, setIsMacOS] = useState(false);
65
67
  const [keyPressed, setKeyPressed] = useState([]);
66
68
  const [isOpen, toggleIsOpen] = useReducer((state) => !state, false);
@@ -106,6 +108,11 @@ const SearchInput = forwardRef(({
106
108
  window.addEventListener("resize", resizeSearchBar);
107
109
  return () => window.removeEventListener("resize", resizeSearchBar);
108
110
  }, []);
111
+ useEffect(() => {
112
+ if (value) {
113
+ setSearchTerms(value);
114
+ }
115
+ }, [value]);
109
116
  const onSearchCallback = (localValue) => {
110
117
  setSearchTerms(localValue);
111
118
  try {
@@ -21,5 +21,5 @@ export type SearchInputProps = {
21
21
  */
22
22
  shortcut?: boolean | ComponentProps<typeof KeyGroup>['keys'];
23
23
  className?: string;
24
- } & Exclude<ComponentProps<typeof TextInputV2>, 'prefix' | 'suffix' | 'clearable' | 'success' | 'onRandomize' | 'onChange' | 'type' | 'value'>;
24
+ } & Exclude<ComponentProps<typeof TextInputV2>, 'prefix' | 'suffix' | 'clearable' | 'success' | 'onRandomize' | 'onChange' | 'type'>;
25
25
  export {};
@@ -83,7 +83,7 @@ const SelectInputProvider = ({
83
83
  }, [options]);
84
84
  const reducer = (state, action) => {
85
85
  switch (action.type) {
86
- case "selectAll": {
86
+ case "selectAll":
87
87
  if (state.allSelected) {
88
88
  return {
89
89
  selectedValues: [],
@@ -96,8 +96,7 @@ const SelectInputProvider = ({
96
96
  allSelected: true,
97
97
  selectedGroups: allGroups
98
98
  };
99
- }
100
- case "selectGroup": {
99
+ case "selectGroup":
101
100
  if (!Array.isArray(options)) {
102
101
  if (state.selectedGroups.includes(action.selectedGroup)) {
103
102
  return {
@@ -115,8 +114,7 @@ const SelectInputProvider = ({
115
114
  };
116
115
  }
117
116
  return state;
118
- }
119
- case "selectOption": {
117
+ case "selectOption":
120
118
  if (multiselect) {
121
119
  if (state.selectedValues.includes(action.clickedOption.value)) {
122
120
  return {
@@ -136,15 +134,13 @@ const SelectInputProvider = ({
136
134
  allSelected: false,
137
135
  selectedGroups: state.selectedGroups
138
136
  };
139
- }
140
- case "clearAll": {
137
+ case "clearAll":
141
138
  return {
142
139
  selectedGroups: [],
143
140
  selectedValues: [],
144
141
  allSelected: false
145
142
  };
146
- }
147
- case "update": {
143
+ case "update":
148
144
  return {
149
145
  selectedGroups: state.selectedGroups,
150
146
  allSelected: state.allSelected,
@@ -155,17 +151,14 @@ const SelectInputProvider = ({
155
151
  return options.some((option) => option.value === selectedValue && !option.disabled);
156
152
  })
157
153
  };
158
- }
159
- case "reset": {
154
+ case "reset":
160
155
  return {
161
156
  selectedValues: action.selectedValues,
162
157
  allSelected: false,
163
158
  selectedGroups: action.selectedGroups
164
159
  };
165
- }
166
- default: {
160
+ default:
167
161
  return state;
168
- }
169
162
  }
170
163
  };
171
164
  const [selectedData, setSelectedData] = React.useReducer(reducer, {
@@ -81,7 +81,7 @@ const SelectInputProvider = ({
81
81
  }, [options]);
82
82
  const reducer = (state, action) => {
83
83
  switch (action.type) {
84
- case "selectAll": {
84
+ case "selectAll":
85
85
  if (state.allSelected) {
86
86
  return {
87
87
  selectedValues: [],
@@ -94,8 +94,7 @@ const SelectInputProvider = ({
94
94
  allSelected: true,
95
95
  selectedGroups: allGroups
96
96
  };
97
- }
98
- case "selectGroup": {
97
+ case "selectGroup":
99
98
  if (!Array.isArray(options)) {
100
99
  if (state.selectedGroups.includes(action.selectedGroup)) {
101
100
  return {
@@ -113,8 +112,7 @@ const SelectInputProvider = ({
113
112
  };
114
113
  }
115
114
  return state;
116
- }
117
- case "selectOption": {
115
+ case "selectOption":
118
116
  if (multiselect) {
119
117
  if (state.selectedValues.includes(action.clickedOption.value)) {
120
118
  return {
@@ -134,15 +132,13 @@ const SelectInputProvider = ({
134
132
  allSelected: false,
135
133
  selectedGroups: state.selectedGroups
136
134
  };
137
- }
138
- case "clearAll": {
135
+ case "clearAll":
139
136
  return {
140
137
  selectedGroups: [],
141
138
  selectedValues: [],
142
139
  allSelected: false
143
140
  };
144
- }
145
- case "update": {
141
+ case "update":
146
142
  return {
147
143
  selectedGroups: state.selectedGroups,
148
144
  allSelected: state.allSelected,
@@ -153,17 +149,14 @@ const SelectInputProvider = ({
153
149
  return options.some((option) => option.value === selectedValue && !option.disabled);
154
150
  })
155
151
  };
156
- }
157
- case "reset": {
152
+ case "reset":
158
153
  return {
159
154
  selectedValues: action.selectedValues,
160
155
  allSelected: false,
161
156
  selectedGroups: action.selectedGroups
162
157
  };
163
- }
164
- default: {
158
+ default:
165
159
  return state;
166
- }
167
160
  }
168
161
  };
169
162
  const [selectedData, setSelectedData] = useReducer(reducer, {
@@ -80,7 +80,7 @@ const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.en
80
80
  theme
81
81
  }) => theme.colors.neutral.textDisabled, ";border:solid 1px ", ({
82
82
  theme
83
- }) => theme.colors.neutral.borderDisabled, ";}" + (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":"AAmCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  color: ${({ theme }) => 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, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => 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 {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\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  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\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  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\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\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        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (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 = structuredClone(vals)\n\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  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\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=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${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      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
83
+ }) => theme.colors.neutral.borderDisabled, ";}" + (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":"AAmCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  color: ${({ theme }) => 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, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => 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 {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\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  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\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  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\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\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\n        case 'ArrowLeft':\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n\n        case 'ArrowRight':\n          event.preventDefault()\n          next?.current?.focus()\n          break\n\n        case 'ArrowUp':\n          event.preventDefault()\n          first?.current?.focus()\n          break\n\n        case 'ArrowDown':\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n\n        default:\n          break\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (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 = structuredClone(vals)\n\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  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\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=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${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      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
84
84
  const FieldSet = /* @__PURE__ */ _styled__default.default("fieldset", process.env.NODE_ENV === "production" ? {
85
85
  target: "e1a2bx9q0"
86
86
  } : {
@@ -88,7 +88,7 @@ const FieldSet = /* @__PURE__ */ _styled__default.default("fieldset", process.en
88
88
  label: "FieldSet"
89
89
  })("border:none;padding:0;margin:0;display:flex;flex-direction:column;gap:", ({
90
90
  theme
91
- }) => theme.space["0.5"], ";" + (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":"AA2GgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  color: ${({ theme }) => 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, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => 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 {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\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  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\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  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\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\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        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (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 = structuredClone(vals)\n\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  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\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=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${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      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
91
+ }) => theme.space["0.5"], ";" + (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":"AA2GgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  color: ${({ theme }) => 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, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => 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 {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\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  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\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  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\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\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\n        case 'ArrowLeft':\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n\n        case 'ArrowRight':\n          event.preventDefault()\n          next?.current?.focus()\n          break\n\n        case 'ArrowUp':\n          event.preventDefault()\n          first?.current?.focus()\n          break\n\n        case 'ArrowDown':\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n\n        default:\n          break\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (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 = structuredClone(vals)\n\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  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\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=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${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      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
92
92
  const DEFAULT_ON_FUNCTION = () => {
93
93
  };
94
94
  const inputOnFocus = (event) => event.target.select();
@@ -158,7 +158,7 @@ const VerificationCode = ({
158
158
  const next = inputRefs[nextIndex];
159
159
  const vals = [...values];
160
160
  switch (event.key) {
161
- case "Backspace": {
161
+ case "Backspace":
162
162
  event.preventDefault();
163
163
  if (values[index2]) {
164
164
  vals[index2] = "";
@@ -171,27 +171,22 @@ const VerificationCode = ({
171
171
  triggerChange(vals);
172
172
  }
173
173
  break;
174
- }
175
- case "ArrowLeft": {
174
+ case "ArrowLeft":
176
175
  event.preventDefault();
177
176
  prev?.current?.focus();
178
177
  break;
179
- }
180
- case "ArrowRight": {
178
+ case "ArrowRight":
181
179
  event.preventDefault();
182
180
  next?.current?.focus();
183
181
  break;
184
- }
185
- case "ArrowUp": {
182
+ case "ArrowUp":
186
183
  event.preventDefault();
187
184
  first?.current?.focus();
188
185
  break;
189
- }
190
- case "ArrowDown": {
186
+ case "ArrowDown":
191
187
  event.preventDefault();
192
188
  last?.current?.focus();
193
189
  break;
194
- }
195
190
  }
196
191
  };
197
192
  const inputOnPaste = (currentIndex) => (event) => {
@@ -228,7 +223,7 @@ const VerificationCode = ({
228
223
  ] }) : null,
229
224
  labelDescription ?? null
230
225
  ] }) : null,
231
- /* @__PURE__ */ jsxRuntime.jsx("div", { children: values.map((value, index2) => /* @__PURE__ */ jsxRuntime.jsx(StyledInput, { 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":"AAsWY","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  color: ${({ theme }) => 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, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => 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 {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\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  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\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  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\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\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        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (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 = structuredClone(vals)\n\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  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\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=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${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      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"], "aria-invalid": !!error, "data-success": !!success, inputSize: size, type: type === "number" ? "tel" : type, pattern: type === "number" ? "[0-9]*" : void 0, "data-testid": index2, value, id: `${inputId || uniqueId}-${index2}`, ref: inputRefs[index2], onChange: inputOnChange(index2), onKeyDown: inputOnKeyDown(index2), onPaste: inputOnPaste(index2), onFocus: inputOnFocus, disabled, required, placeholder: placeholder?.[index2] ?? "", "aria-label": `${ariaLabel} ${index2}` }, `field-${index2}`)) }),
226
+ /* @__PURE__ */ jsxRuntime.jsx("div", { children: values.map((value, index2) => /* @__PURE__ */ jsxRuntime.jsx(StyledInput, { 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":"AAmWY","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  color: ${({ theme }) => 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, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => 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 {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\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  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\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  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\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\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\n        case 'ArrowLeft':\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n\n        case 'ArrowRight':\n          event.preventDefault()\n          next?.current?.focus()\n          break\n\n        case 'ArrowUp':\n          event.preventDefault()\n          first?.current?.focus()\n          break\n\n        case 'ArrowDown':\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n\n        default:\n          break\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (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 = structuredClone(vals)\n\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  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\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=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${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      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"], "aria-invalid": !!error, "data-success": !!success, inputSize: size, type: type === "number" ? "tel" : type, pattern: type === "number" ? "[0-9]*" : void 0, "data-testid": index2, value, id: `${inputId || uniqueId}-${index2}`, ref: inputRefs[index2], onChange: inputOnChange(index2), onKeyDown: inputOnKeyDown(index2), onPaste: inputOnPaste(index2), onFocus: inputOnFocus, disabled, required, placeholder: placeholder?.[index2] ?? "", "aria-label": `${ariaLabel} ${index2}` }, `field-${index2}`)) }),
232
227
  error || typeof success === "string" || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", variant: "caption", sentiment, prominence: !error && !success ? "weak" : "default", disabled, children: error || success || helper }) : null,
233
228
  !error && !success && typeof helper !== "string" && helper ? helper : null
234
229
  ] });