@ultraviolet/ui 1.58.0 → 1.59.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.
@@ -4,6 +4,7 @@ const jsxRuntime = require("@emotion/react/jsx-runtime");
4
4
  const _styled = require("@emotion/styled/base");
5
5
  const icons = require("@ultraviolet/icons");
6
6
  const React = require("react");
7
+ const isClientSide = require("../../helpers/isClientSide.cjs");
7
8
  const index = require("../Popup/index.cjs");
8
9
  const index$1 = require("../TextInputV2/index.cjs");
9
10
  const KeyGroup = require("./KeyGroup.cjs");
@@ -20,13 +21,13 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
20
21
  theme
21
22
  }) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
22
23
  theme
23
- }) => `${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":"AAsBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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 [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 = (value: string) => {\n      setSearchTerms(value)\n\n      try {\n        onSearch(value)\n        if (value.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    const isMacOS = navigator.userAgent.includes('Mac')\n\n    const handleShortcut = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n\n        if (\n          (key === 'k' || key === 'K') &&\n          ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n        ) {\n          event.preventDefault()\n          innerSearchInputRef.current?.focus()\n        }\n      },\n      [isMacOS, innerSearchInputRef],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleShortcut)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleShortcut)\n      }\n    }, [handleShortcut, shortcut, disabled])\n\n    return (\n      <div>\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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup\n                  disabled={disabled}\n                  keys={[isMacOS ? '⌘' : 'Ctrl', 'K']}\n                />\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={onSearchCallback}\n            clearable\n            disabled={disabled}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
24
+ }) => `${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 { Icon } 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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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        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        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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\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={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
24
25
  const StyledTextInputV2 = /* @__PURE__ */ _styled__default.default(index$1.TextInputV2, process.env.NODE_ENV === "production" ? {
25
26
  target: "eefux4u0"
26
27
  } : {
27
28
  target: "eefux4u0",
28
29
  label: "StyledTextInputV2"
29
- })(index$1.BasicPrefixStack, "{border:none;}", index$1.StyledInput, "{padding:0;}", index$1.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":"AA+B6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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 [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 = (value: string) => {\n      setSearchTerms(value)\n\n      try {\n        onSearch(value)\n        if (value.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    const isMacOS = navigator.userAgent.includes('Mac')\n\n    const handleShortcut = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n\n        if (\n          (key === 'k' || key === 'K') &&\n          ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n        ) {\n          event.preventDefault()\n          innerSearchInputRef.current?.focus()\n        }\n      },\n      [isMacOS, innerSearchInputRef],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleShortcut)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleShortcut)\n      }\n    }, [handleShortcut, shortcut, disabled])\n\n    return (\n      <div>\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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup\n                  disabled={disabled}\n                  keys={[isMacOS ? '⌘' : 'Ctrl', 'K']}\n                />\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={onSearchCallback}\n            clearable\n            disabled={disabled}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
30
+ })(index$1.BasicPrefixStack, "{border:none;}", index$1.StyledInput, "{padding:0;}", index$1.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 { Icon } 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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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        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        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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\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={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
30
31
  const SearchInput = React.forwardRef(({
31
32
  placeholder,
32
33
  label,
@@ -40,12 +41,15 @@ const SearchInput = React.forwardRef(({
40
41
  "data-testid": dataTestId,
41
42
  shortcut = false,
42
43
  error,
43
- disabled
44
+ disabled,
45
+ className
44
46
  }, ref) => {
45
47
  const focusedLinkIndex = React.useRef(0);
46
48
  const popupRef = React.useRef(null);
47
49
  const [containerWidth, setContainerWidth] = React.useState(0);
48
50
  const [searchTerms, setSearchTerms] = React.useState("");
51
+ const [isMacOS, setIsMacOS] = React.useState(false);
52
+ const [keyPressed, setKeyPressed] = React.useState([]);
49
53
  const [isOpen, toggleIsOpen] = React.useReducer((state) => !state, false);
50
54
  const innerSearchInputRef = React.useRef(null);
51
55
  React.useImperativeHandle(ref, () => innerSearchInputRef.current);
@@ -89,11 +93,11 @@ const SearchInput = React.forwardRef(({
89
93
  window.addEventListener("resize", resizeSearchBar);
90
94
  return () => window.removeEventListener("resize", resizeSearchBar);
91
95
  }, []);
92
- const onSearchCallback = (value) => {
93
- setSearchTerms(value);
96
+ const onSearchCallback = (localValue) => {
97
+ setSearchTerms(localValue);
94
98
  try {
95
- onSearch(value);
96
- if (value.length >= threshold && !isOpen) {
99
+ onSearch(localValue);
100
+ if (localValue.length >= threshold && !isOpen) {
97
101
  toggleIsOpen();
98
102
  }
99
103
  } catch {
@@ -106,26 +110,64 @@ const SearchInput = React.forwardRef(({
106
110
  toggleIsOpen();
107
111
  }
108
112
  };
109
- const isMacOS = navigator.userAgent.includes("Mac");
110
- const handleShortcut = React.useCallback((event) => {
113
+ React.useEffect(() => {
114
+ if (isClientSide.isClientSide) {
115
+ setIsMacOS(navigator.userAgent.includes("Mac"));
116
+ }
117
+ }, []);
118
+ const handleKeyPressed = React.useCallback((event) => {
111
119
  const {
112
120
  ctrlKey,
113
121
  metaKey,
114
122
  key
115
123
  } = event;
116
- if ((key === "k" || key === "K") && (!isMacOS && ctrlKey || isMacOS && metaKey)) {
117
- event.preventDefault();
118
- innerSearchInputRef.current?.focus();
124
+ setKeyPressed([...keyPressed, key.toUpperCase()]);
125
+ if (typeof shortcut === "boolean") {
126
+ if ((key === "k" || key === "K") && (!isMacOS && ctrlKey || isMacOS && metaKey)) {
127
+ event.preventDefault();
128
+ innerSearchInputRef.current?.focus();
129
+ }
130
+ } else {
131
+ const uppercaseShortcut = shortcut.map((s) => s.toUpperCase());
132
+ if (JSON.stringify([...keyPressed, key.toUpperCase()]) === JSON.stringify(uppercaseShortcut)) {
133
+ event.preventDefault();
134
+ innerSearchInputRef.current?.focus();
135
+ }
119
136
  }
120
- }, [isMacOS, innerSearchInputRef]);
137
+ }, [keyPressed, shortcut, isMacOS]);
138
+ const handleKeyReleased = React.useCallback((event) => {
139
+ const {
140
+ key
141
+ } = event;
142
+ setKeyPressed(keyPressed.filter((k) => k !== key.toUpperCase()));
143
+ }, [keyPressed]);
121
144
  React.useEffect(() => {
122
145
  if (shortcut && !disabled) {
123
- document.body.addEventListener("keydown", handleShortcut);
146
+ document.body.addEventListener("keydown", handleKeyPressed);
147
+ document.body.addEventListener("keyup", handleKeyReleased);
124
148
  }
125
149
  return () => {
126
- document.body.removeEventListener("keydown", handleShortcut);
150
+ document.body.removeEventListener("keydown", handleKeyPressed);
151
+ document.body.removeEventListener("keyup", handleKeyReleased);
127
152
  };
128
- }, [handleShortcut, shortcut, disabled]);
129
- return /* @__PURE__ */ jsxRuntime.jsx("div", { children: /* @__PURE__ */ jsxRuntime.jsx(StyledPopup, { "data-testid": `popup-${dataTestId}`, role: "dialog", visible: isOpen, onClose: onCloseCallback, placement: popupPlacement, maxWidth: containerWidth, hideOnClickOutside: true, hasArrow: false, innerRef: popupRef, text: content, maxHeight: 410, debounceDelay: 0, children: /* @__PURE__ */ jsxRuntime.jsx(StyledTextInputV2, { ref: innerSearchInputRef, prefix: /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "search", disabled, sentiment: "neutral" }), suffix: shortcut && searchTerms.length === 0 ? /* @__PURE__ */ jsxRuntime.jsx(KeyGroup.KeyGroup, { disabled, keys: [isMacOS ? "⌘" : "Ctrl", "K"] }) : void 0, "data-testid": dataTestId, error, value: searchTerms, size, label, placeholder, loading, onChange: onSearchCallback, clearable: true, disabled }) }) });
153
+ }, [shortcut, disabled, handleKeyPressed, handleKeyReleased]);
154
+ const keys = React.useMemo(() => {
155
+ if (typeof shortcut === "boolean") {
156
+ return [isMacOS ? "⌘" : "Ctrl", "K"];
157
+ }
158
+ const filteredKey = shortcut.map((key) => {
159
+ if (key === "Meta") {
160
+ return "⌘";
161
+ }
162
+ if (key === "Control") {
163
+ return "Ctrl";
164
+ }
165
+ return key;
166
+ });
167
+ return filteredKey;
168
+ }, [isMacOS, shortcut]);
169
+ return /* @__PURE__ */ jsxRuntime.jsx("div", { style: {
170
+ width: "100%"
171
+ }, children: /* @__PURE__ */ jsxRuntime.jsx(StyledPopup, { "data-testid": `popup-${dataTestId}`, role: "dialog", visible: isOpen, onClose: onCloseCallback, placement: popupPlacement, maxWidth: containerWidth, hideOnClickOutside: true, hasArrow: false, innerRef: popupRef, text: content, maxHeight: 410, debounceDelay: 0, children: /* @__PURE__ */ jsxRuntime.jsx(StyledTextInputV2, { ref: innerSearchInputRef, prefix: /* @__PURE__ */ jsxRuntime.jsx(icons.Icon, { name: "search", disabled, sentiment: "neutral" }), suffix: shortcut && searchTerms.length === 0 ? /* @__PURE__ */ jsxRuntime.jsx(KeyGroup.KeyGroup, { disabled, keys }) : void 0, "data-testid": dataTestId, error, value: searchTerms, size, label, placeholder, loading, onChange: onSearchCallback, clearable: true, disabled, className }) }) });
130
172
  });
131
173
  exports.SearchInput = SearchInput;
@@ -1,4 +1,5 @@
1
1
  import { Popup } from '../Popup';
2
+ import { KeyGroup } from './KeyGroup';
2
3
  /**
3
4
  * 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.
4
5
  * Children of the SearchInput component can be a function that receives an object with the following properties:
@@ -9,11 +10,12 @@ import { Popup } from '../Popup';
9
10
  export declare const SearchInput: import("react").ForwardRefExoticComponent<{
10
11
  popupPlacement?: import("react").ComponentProps<typeof Popup>["placement"];
11
12
  threshold?: number;
12
- children: import("./types").SearchBarChildrenFunctionProps | import("react").ReactNode;
13
+ children?: import("./types").SearchBarChildrenFunctionProps | import("react").ReactNode;
13
14
  onSearch: (value: string) => void;
14
15
  onClose?: () => void;
15
16
  "data-testid"?: string;
16
- shortcut?: boolean;
17
+ shortcut?: boolean | import("react").ComponentProps<typeof KeyGroup>["keys"];
18
+ className?: string;
17
19
  } & Pick<{
18
20
  className?: string;
19
21
  clearable?: boolean;
@@ -34,4 +36,4 @@ export declare const SearchInput: import("react").ForwardRefExoticComponent<{
34
36
  tooltip?: string;
35
37
  type?: "text" | "password" | "url" | "email";
36
38
  value?: string;
37
- } & Pick<import("react").InputHTMLAttributes<HTMLInputElement>, "autoFocus" | "id" | "tabIndex" | "aria-label" | "aria-labelledby" | "onFocus" | "onBlur" | "onKeyDown" | "name" | "disabled" | "autoComplete" | "placeholder" | "readOnly" | "required"> & import("react").RefAttributes<HTMLInputElement>, "label" | "size" | "error" | "disabled" | "loading" | "placeholder"> & import("react").RefAttributes<HTMLInputElement>>;
39
+ } & Pick<import("react").InputHTMLAttributes<HTMLInputElement>, "role" | "autoFocus" | "id" | "tabIndex" | "aria-atomic" | "aria-label" | "aria-labelledby" | "aria-live" | "onFocus" | "onBlur" | "onKeyDown" | "name" | "disabled" | "autoComplete" | "placeholder" | "readOnly" | "required"> & import("react").RefAttributes<HTMLInputElement>, "label" | "size" | "error" | "disabled" | "loading" | "placeholder"> & import("react").RefAttributes<HTMLInputElement>>;
@@ -1,7 +1,8 @@
1
1
  import { jsx } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
3
  import { Icon } from "@ultraviolet/icons";
4
- import { forwardRef, useRef, useState, useReducer, useImperativeHandle, useEffect, useCallback } from "react";
4
+ import { forwardRef, useRef, useState, useReducer, useImperativeHandle, useEffect, useCallback, useMemo } from "react";
5
+ import { isClientSide } from "../../helpers/isClientSide.js";
5
6
  import { Popup } from "../Popup/index.js";
6
7
  import { TextInputV2, BasicPrefixStack, StyledInput, BasicSuffixStack } from "../TextInputV2/index.js";
7
8
  import { KeyGroup } from "./KeyGroup.js";
@@ -16,13 +17,13 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
16
17
  theme
17
18
  }) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
18
19
  theme
19
- }) => `${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":"AAsBiC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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 [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 = (value: string) => {\n      setSearchTerms(value)\n\n      try {\n        onSearch(value)\n        if (value.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    const isMacOS = navigator.userAgent.includes('Mac')\n\n    const handleShortcut = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n\n        if (\n          (key === 'k' || key === 'K') &&\n          ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n        ) {\n          event.preventDefault()\n          innerSearchInputRef.current?.focus()\n        }\n      },\n      [isMacOS, innerSearchInputRef],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleShortcut)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleShortcut)\n      }\n    }, [handleShortcut, shortcut, disabled])\n\n    return (\n      <div>\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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup\n                  disabled={disabled}\n                  keys={[isMacOS ? '⌘' : 'Ctrl', 'K']}\n                />\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={onSearchCallback}\n            clearable\n            disabled={disabled}\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 { Icon } 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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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        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        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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\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={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
20
21
  const StyledTextInputV2 = /* @__PURE__ */ _styled(TextInputV2, process.env.NODE_ENV === "production" ? {
21
22
  target: "eefux4u0"
22
23
  } : {
23
24
  target: "eefux4u0",
24
25
  label: "StyledTextInputV2"
25
- })(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":"AA+B6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/SearchInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Icon } from '@ultraviolet/icons'\nimport type { Ref } from 'react'\nimport {\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useReducer,\n  useRef,\n  useState,\n} from 'react'\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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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 [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 = (value: string) => {\n      setSearchTerms(value)\n\n      try {\n        onSearch(value)\n        if (value.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    const isMacOS = navigator.userAgent.includes('Mac')\n\n    const handleShortcut = useCallback(\n      (event: KeyboardEvent) => {\n        const { ctrlKey, metaKey, key } = event\n\n        if (\n          (key === 'k' || key === 'K') &&\n          ((!isMacOS && ctrlKey) || (isMacOS && metaKey))\n        ) {\n          event.preventDefault()\n          innerSearchInputRef.current?.focus()\n        }\n      },\n      [isMacOS, innerSearchInputRef],\n    )\n\n    useEffect(() => {\n      if (shortcut && !disabled) {\n        document.body.addEventListener('keydown', handleShortcut)\n      }\n\n      return () => {\n        document.body.removeEventListener('keydown', handleShortcut)\n      }\n    }, [handleShortcut, shortcut, disabled])\n\n    return (\n      <div>\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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\n            suffix={\n              shortcut && searchTerms.length === 0 ? (\n                <KeyGroup\n                  disabled={disabled}\n                  keys={[isMacOS ? '⌘' : 'Ctrl', 'K']}\n                />\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={onSearchCallback}\n            clearable\n            disabled={disabled}\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 { Icon } 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: 610px;\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(\n  (\n    {\n      placeholder,\n      label,\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    }: 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        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        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={\n              <Icon name=\"search\" disabled={disabled} sentiment=\"neutral\" />\n            }\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={onSearchCallback}\n            clearable\n            disabled={disabled}\n            className={className}\n          />\n        </StyledPopup>\n      </div>\n    )\n  },\n)\n"]} */"));
26
27
  const SearchInput = forwardRef(({
27
28
  placeholder,
28
29
  label,
@@ -36,12 +37,15 @@ const SearchInput = forwardRef(({
36
37
  "data-testid": dataTestId,
37
38
  shortcut = false,
38
39
  error,
39
- disabled
40
+ disabled,
41
+ className
40
42
  }, ref) => {
41
43
  const focusedLinkIndex = useRef(0);
42
44
  const popupRef = useRef(null);
43
45
  const [containerWidth, setContainerWidth] = useState(0);
44
46
  const [searchTerms, setSearchTerms] = useState("");
47
+ const [isMacOS, setIsMacOS] = useState(false);
48
+ const [keyPressed, setKeyPressed] = useState([]);
45
49
  const [isOpen, toggleIsOpen] = useReducer((state) => !state, false);
46
50
  const innerSearchInputRef = useRef(null);
47
51
  useImperativeHandle(ref, () => innerSearchInputRef.current);
@@ -85,11 +89,11 @@ const SearchInput = forwardRef(({
85
89
  window.addEventListener("resize", resizeSearchBar);
86
90
  return () => window.removeEventListener("resize", resizeSearchBar);
87
91
  }, []);
88
- const onSearchCallback = (value) => {
89
- setSearchTerms(value);
92
+ const onSearchCallback = (localValue) => {
93
+ setSearchTerms(localValue);
90
94
  try {
91
- onSearch(value);
92
- if (value.length >= threshold && !isOpen) {
95
+ onSearch(localValue);
96
+ if (localValue.length >= threshold && !isOpen) {
93
97
  toggleIsOpen();
94
98
  }
95
99
  } catch {
@@ -102,27 +106,65 @@ const SearchInput = forwardRef(({
102
106
  toggleIsOpen();
103
107
  }
104
108
  };
105
- const isMacOS = navigator.userAgent.includes("Mac");
106
- const handleShortcut = useCallback((event) => {
109
+ useEffect(() => {
110
+ if (isClientSide) {
111
+ setIsMacOS(navigator.userAgent.includes("Mac"));
112
+ }
113
+ }, []);
114
+ const handleKeyPressed = useCallback((event) => {
107
115
  const {
108
116
  ctrlKey,
109
117
  metaKey,
110
118
  key
111
119
  } = event;
112
- if ((key === "k" || key === "K") && (!isMacOS && ctrlKey || isMacOS && metaKey)) {
113
- event.preventDefault();
114
- innerSearchInputRef.current?.focus();
120
+ setKeyPressed([...keyPressed, key.toUpperCase()]);
121
+ if (typeof shortcut === "boolean") {
122
+ if ((key === "k" || key === "K") && (!isMacOS && ctrlKey || isMacOS && metaKey)) {
123
+ event.preventDefault();
124
+ innerSearchInputRef.current?.focus();
125
+ }
126
+ } else {
127
+ const uppercaseShortcut = shortcut.map((s) => s.toUpperCase());
128
+ if (JSON.stringify([...keyPressed, key.toUpperCase()]) === JSON.stringify(uppercaseShortcut)) {
129
+ event.preventDefault();
130
+ innerSearchInputRef.current?.focus();
131
+ }
115
132
  }
116
- }, [isMacOS, innerSearchInputRef]);
133
+ }, [keyPressed, shortcut, isMacOS]);
134
+ const handleKeyReleased = useCallback((event) => {
135
+ const {
136
+ key
137
+ } = event;
138
+ setKeyPressed(keyPressed.filter((k) => k !== key.toUpperCase()));
139
+ }, [keyPressed]);
117
140
  useEffect(() => {
118
141
  if (shortcut && !disabled) {
119
- document.body.addEventListener("keydown", handleShortcut);
142
+ document.body.addEventListener("keydown", handleKeyPressed);
143
+ document.body.addEventListener("keyup", handleKeyReleased);
120
144
  }
121
145
  return () => {
122
- document.body.removeEventListener("keydown", handleShortcut);
146
+ document.body.removeEventListener("keydown", handleKeyPressed);
147
+ document.body.removeEventListener("keyup", handleKeyReleased);
123
148
  };
124
- }, [handleShortcut, shortcut, disabled]);
125
- return /* @__PURE__ */ jsx("div", { children: /* @__PURE__ */ jsx(StyledPopup, { "data-testid": `popup-${dataTestId}`, role: "dialog", visible: isOpen, onClose: onCloseCallback, placement: popupPlacement, maxWidth: containerWidth, hideOnClickOutside: true, hasArrow: false, innerRef: popupRef, text: content, maxHeight: 410, debounceDelay: 0, children: /* @__PURE__ */ jsx(StyledTextInputV2, { ref: innerSearchInputRef, prefix: /* @__PURE__ */ jsx(Icon, { name: "search", disabled, sentiment: "neutral" }), suffix: shortcut && searchTerms.length === 0 ? /* @__PURE__ */ jsx(KeyGroup, { disabled, keys: [isMacOS ? "⌘" : "Ctrl", "K"] }) : void 0, "data-testid": dataTestId, error, value: searchTerms, size, label, placeholder, loading, onChange: onSearchCallback, clearable: true, disabled }) }) });
149
+ }, [shortcut, disabled, handleKeyPressed, handleKeyReleased]);
150
+ const keys = useMemo(() => {
151
+ if (typeof shortcut === "boolean") {
152
+ return [isMacOS ? "⌘" : "Ctrl", "K"];
153
+ }
154
+ const filteredKey = shortcut.map((key) => {
155
+ if (key === "Meta") {
156
+ return "⌘";
157
+ }
158
+ if (key === "Control") {
159
+ return "Ctrl";
160
+ }
161
+ return key;
162
+ });
163
+ return filteredKey;
164
+ }, [isMacOS, shortcut]);
165
+ return /* @__PURE__ */ jsx("div", { style: {
166
+ width: "100%"
167
+ }, children: /* @__PURE__ */ jsx(StyledPopup, { "data-testid": `popup-${dataTestId}`, role: "dialog", visible: isOpen, onClose: onCloseCallback, placement: popupPlacement, maxWidth: containerWidth, hideOnClickOutside: true, hasArrow: false, innerRef: popupRef, text: content, maxHeight: 410, debounceDelay: 0, children: /* @__PURE__ */ jsx(StyledTextInputV2, { ref: innerSearchInputRef, prefix: /* @__PURE__ */ jsx(Icon, { name: "search", disabled, sentiment: "neutral" }), suffix: shortcut && searchTerms.length === 0 ? /* @__PURE__ */ jsx(KeyGroup, { disabled, keys }) : void 0, "data-testid": dataTestId, error, value: searchTerms, size, label, placeholder, loading, onChange: onSearchCallback, clearable: true, disabled, className }) }) });
126
168
  });
127
169
  export {
128
170
  SearchInput
@@ -1,5 +1,6 @@
1
1
  import type { Popup, TextInputV2 } from '@ultraviolet/ui';
2
2
  import type { ComponentProps, DispatchWithoutAction, ReactNode } from 'react';
3
+ import type { KeyGroup } from './KeyGroup';
3
4
  type ChildrenProps = {
4
5
  searchTerms: string;
5
6
  isOpen: boolean;
@@ -9,13 +10,15 @@ export type SearchBarChildrenFunctionProps = ({ searchTerms, isOpen, toggleIsOpe
9
10
  export type SearchInputProps = {
10
11
  popupPlacement?: ComponentProps<typeof Popup>['placement'];
11
12
  threshold?: number;
12
- children: SearchBarChildrenFunctionProps | ReactNode;
13
+ children?: SearchBarChildrenFunctionProps | ReactNode;
13
14
  onSearch: (value: string) => void;
14
15
  onClose?: () => void;
15
16
  ['data-testid']?: string;
16
17
  /**
17
- * If set to true images will be shown with key shortcut to focus the input on the right of the search bar
18
+ * If set to true images will be shown with key shortcut to focus the input on the right of the search bar.
19
+ * If set to an array of strings, the strings will be used as the key shortcuts.
18
20
  */
19
- shortcut?: boolean;
21
+ shortcut?: boolean | ComponentProps<typeof KeyGroup>['keys'];
22
+ className?: string;
20
23
  } & Pick<ComponentProps<typeof TextInputV2>, 'placeholder' | 'size' | 'label' | 'loading' | 'error' | 'disabled'>;
21
24
  export {};