@ultraviolet/ui 1.67.1 → 1.67.2
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.
|
@@ -21,13 +21,13 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
|
|
|
21
21
|
theme
|
|
22
22
|
}) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
|
|
23
23
|
theme
|
|
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      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        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={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"]} */"));
|
|
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      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={\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={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"]} */"));
|
|
25
25
|
const StyledTextInputV2 = /* @__PURE__ */ _styled__default.default(index$1.TextInputV2, process.env.NODE_ENV === "production" ? {
|
|
26
26
|
target: "eefux4u0"
|
|
27
27
|
} : {
|
|
28
28
|
target: "eefux4u0",
|
|
29
29
|
label: "StyledTextInputV2"
|
|
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      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        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={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"]} */"));
|
|
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      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={\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={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"]} */"));
|
|
31
31
|
const SearchInput = React.forwardRef(({
|
|
32
32
|
placeholder,
|
|
33
33
|
label,
|
|
@@ -156,6 +156,9 @@ const SearchInput = React.forwardRef(({
|
|
|
156
156
|
}
|
|
157
157
|
}, [keyPressed, shortcut, isMacOS]);
|
|
158
158
|
const handleKeyReleased = React.useCallback((event) => {
|
|
159
|
+
if (!(event instanceof KeyboardEvent)) {
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
159
162
|
const {
|
|
160
163
|
key
|
|
161
164
|
} = event;
|
|
@@ -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 { 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      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        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={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 { 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      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={\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={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 { 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      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        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={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 { 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      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={\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={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,
|
|
@@ -152,6 +152,9 @@ const SearchInput = forwardRef(({
|
|
|
152
152
|
}
|
|
153
153
|
}, [keyPressed, shortcut, isMacOS]);
|
|
154
154
|
const handleKeyReleased = useCallback((event) => {
|
|
155
|
+
if (!(event instanceof KeyboardEvent)) {
|
|
156
|
+
return;
|
|
157
|
+
}
|
|
155
158
|
const {
|
|
156
159
|
key
|
|
157
160
|
} = event;
|
|
@@ -33,7 +33,7 @@ const StyledTable = /* @__PURE__ */ _styled__default.default("table", process.en
|
|
|
33
33
|
${bordered ? `& tbody tr {
|
|
34
34
|
border-bottom: 1px solid ${theme.colors.neutral.borderWeak};
|
|
35
35
|
}` : ""}
|
|
36
|
-
`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFtQm9CIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcywgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBmb3J3YXJkUmVmIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBCb2R5IH0gZnJvbSAnLi9Cb2R5J1xuaW1wb3J0IHsgQ2VsbCB9IGZyb20gJy4vQ2VsbCdcbmltcG9ydCB7IEhlYWRlciB9IGZyb20gJy4vSGVhZGVyJ1xuaW1wb3J0IHsgSGVhZGVyQ2VsbCB9IGZyb20gJy4vSGVhZGVyQ2VsbCdcbmltcG9ydCB7IEhlYWRlclJvdyB9IGZyb20gJy4vSGVhZGVyUm93J1xuaW1wb3J0IHsgUm93IH0gZnJvbSAnLi9Sb3cnXG5pbXBvcnQgeyBTZWxlY3RCYXIgfSBmcm9tICcuL1NlbGVjdEJhcidcbmltcG9ydCB7IFNrZWxldG9uUm93cyB9IGZyb20gJy4vU2tlbGV0b25Sb3dzJ1xuaW1wb3J0IHsgVGFibGVQcm92aWRlciwgdXNlVGFibGVDb250ZXh0IH0gZnJvbSAnLi9UYWJsZUNvbnRleHQnXG5cbnR5cGUgU3R5bGVkVGFibGVQcm9wcyA9IHtcbiAgc3RyaXBwZWQ6IGJvb2xlYW5cbiAgYm9yZGVyZWQ6IGJvb2xlYW5cbn1cbmNvbnN0IFN0eWxlZFRhYmxlID0gc3R5bGVkKCd0YWJsZScsIHtcbiAgc2hvdWxkRm9yd2FyZFByb3A6IHByb3AgPT4gIVsnYm9yZGVyZWQnLCAnc3RyaXBwZWQnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPFN0eWxlZFRhYmxlUHJvcHM+YFxuICB3aWR0aDogMTAwJTtcbiAgYm94LXNpemluZzogY29udGVudC1ib3g7XG4gIGJvcmRlci1jb2xsYXBzZTogY29sbGFwc2U7XG5cbiAgJHsoeyB0aGVtZSwgc3RyaXBwZWQsIGJvcmRlcmVkIH0pID0+IGBcbiAgJHtcbiAgICBzdHJpcHBlZFxuICAgICAgPyBgJiB0Ym9keSB0cjpudGgtb2YtdHlwZShldmVuKSB7XG4gICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kV2Vha307XG4gIH1gXG4gICAgICA6ICcnXG4gIH1cblxuICAke1xuICAgIGJvcmRlcmVkXG4gICAgICA/IGAmIHRib2R5IHRyIHtcbiAgICBib3JkZXItYm90dG9tOiAxcHggc29saWQgJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5ib3JkZXJXZWFrfTtcbiAgfWBcbiAgICAgIDogJydcbiAgfVxuICBgfVxuYFxuXG50eXBlIENvbHVtblByb3BzID0gUGljazxcbiAgQ29tcG9uZW50UHJvcHM8dHlwZW9mIEhlYWRlckNlbGw+
|
|
36
|
+
`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFtQm9CIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcywgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBmb3J3YXJkUmVmIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBCb2R5IH0gZnJvbSAnLi9Cb2R5J1xuaW1wb3J0IHsgQ2VsbCB9IGZyb20gJy4vQ2VsbCdcbmltcG9ydCB7IEhlYWRlciB9IGZyb20gJy4vSGVhZGVyJ1xuaW1wb3J0IHsgSGVhZGVyQ2VsbCB9IGZyb20gJy4vSGVhZGVyQ2VsbCdcbmltcG9ydCB7IEhlYWRlclJvdyB9IGZyb20gJy4vSGVhZGVyUm93J1xuaW1wb3J0IHsgUm93IH0gZnJvbSAnLi9Sb3cnXG5pbXBvcnQgeyBTZWxlY3RCYXIgfSBmcm9tICcuL1NlbGVjdEJhcidcbmltcG9ydCB7IFNrZWxldG9uUm93cyB9IGZyb20gJy4vU2tlbGV0b25Sb3dzJ1xuaW1wb3J0IHsgVGFibGVQcm92aWRlciwgdXNlVGFibGVDb250ZXh0IH0gZnJvbSAnLi9UYWJsZUNvbnRleHQnXG5cbnR5cGUgU3R5bGVkVGFibGVQcm9wcyA9IHtcbiAgc3RyaXBwZWQ6IGJvb2xlYW5cbiAgYm9yZGVyZWQ6IGJvb2xlYW5cbn1cbmNvbnN0IFN0eWxlZFRhYmxlID0gc3R5bGVkKCd0YWJsZScsIHtcbiAgc2hvdWxkRm9yd2FyZFByb3A6IHByb3AgPT4gIVsnYm9yZGVyZWQnLCAnc3RyaXBwZWQnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPFN0eWxlZFRhYmxlUHJvcHM+YFxuICB3aWR0aDogMTAwJTtcbiAgYm94LXNpemluZzogY29udGVudC1ib3g7XG4gIGJvcmRlci1jb2xsYXBzZTogY29sbGFwc2U7XG5cbiAgJHsoeyB0aGVtZSwgc3RyaXBwZWQsIGJvcmRlcmVkIH0pID0+IGBcbiAgJHtcbiAgICBzdHJpcHBlZFxuICAgICAgPyBgJiB0Ym9keSB0cjpudGgtb2YtdHlwZShldmVuKSB7XG4gICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kV2Vha307XG4gIH1gXG4gICAgICA6ICcnXG4gIH1cblxuICAke1xuICAgIGJvcmRlcmVkXG4gICAgICA/IGAmIHRib2R5IHRyIHtcbiAgICBib3JkZXItYm90dG9tOiAxcHggc29saWQgJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5ib3JkZXJXZWFrfTtcbiAgfWBcbiAgICAgIDogJydcbiAgfVxuICBgfVxuYFxuXG50eXBlIENvbHVtblByb3BzID0gUGljazxcbiAgQ29tcG9uZW50UHJvcHM8dHlwZW9mIEhlYWRlckNlbGw+LFxuICAnaXNPcmRlcmVkJyB8ICdvbk9yZGVyJyB8ICdvcmRlckRpcmVjdGlvbidcbj4gJiB7XG4gIGxhYmVsPzogUmVhY3ROb2RlXG4gIHdpZHRoPzogc3RyaW5nXG4gIG1pbldpZHRoPzogc3RyaW5nXG4gIG1heFdpZHRoPzogc3RyaW5nXG4gIGluZm8/OiBzdHJpbmdcbn1cblxudHlwZSBUYWJsZVByb3BzID0ge1xuICBzZWxlY3RhYmxlPzogYm9vbGVhblxuICBjb2x1bW5zOiBDb2x1bW5Qcm9wc1tdXG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbiAgLyoqXG4gICAqIFNldCBpdCB0byB0cnVlIGlmIHlvdSB3YW50IHRvIGRpc3BsYXkgYSBwbGFjZWhvbGRlciBkdXJpbmcgbG9hZGluZ1xuICAgKiAqL1xuICBsb2FkaW5nPzogYm9vbGVhblxuICBib3JkZXJlZD86IGJvb2xlYW5cbiAgc3RyaXBwZWQ/OiBib29sZWFuXG59XG5cbmV4cG9ydCBjb25zdCBCYXNlVGFibGUgPSBmb3J3YXJkUmVmPEhUTUxUYWJsZUVsZW1lbnQsIFRhYmxlUHJvcHM+KFxuICAoXG4gICAge1xuICAgICAgc2VsZWN0YWJsZSA9IGZhbHNlLFxuICAgICAgY2hpbGRyZW4sXG4gICAgICBjb2x1bW5zLFxuICAgICAgbG9hZGluZyxcbiAgICAgIGJvcmRlcmVkID0gZmFsc2UsXG4gICAgICBzdHJpcHBlZCA9IGZhbHNlLFxuICAgIH0sXG4gICAgcmVmLFxuICApID0+IChcbiAgICA8VGFibGVQcm92aWRlclxuICAgICAgc2VsZWN0YWJsZT17c2VsZWN0YWJsZX1cbiAgICAgIHN0cmlwcGVkPXtzdHJpcHBlZH1cbiAgICAgIGJvcmRlcmVkPXtib3JkZXJlZH1cbiAgICA+XG4gICAgICA8U3R5bGVkVGFibGUgcmVmPXtyZWZ9IHN0cmlwcGVkPXtzdHJpcHBlZH0gYm9yZGVyZWQ9e2JvcmRlcmVkfT5cbiAgICAgICAgPEhlYWRlcj5cbiAgICAgICAgICA8SGVhZGVyUm93IGhhc1NlbGVjdEFsbENvbHVtbj17c2VsZWN0YWJsZX0+XG4gICAgICAgICAgICB7Y29sdW1ucy5tYXAoKGNvbHVtbiwgaW5kZXgpID0+IChcbiAgICAgICAgICAgICAgPEhlYWRlckNlbGxcbiAgICAgICAgICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcmVhY3Qvbm8tYXJyYXktaW5kZXgta2V5XG4gICAgICAgICAgICAgICAga2V5PXtgaGVhZGVyLWNvbHVtbi0ke2luZGV4fWB9XG4gICAgICAgICAgICAgICAgaXNPcmRlcmVkPXtjb2x1bW4uaXNPcmRlcmVkfVxuICAgICAgICAgICAgICAgIG9yZGVyRGlyZWN0aW9uPXtjb2x1bW4ub3JkZXJEaXJlY3Rpb259XG4gICAgICAgICAgICAgICAgb25PcmRlcj17Y29sdW1uLm9uT3JkZXJ9XG4gICAgICAgICAgICAgICAgd2lkdGg9e2NvbHVtbi53aWR0aH1cbiAgICAgICAgICAgICAgICBtaW5XaWR0aD17Y29sdW1uLm1pbldpZHRofVxuICAgICAgICAgICAgICAgIG1heFdpZHRoPXtjb2x1bW4ubWF4V2lkdGh9XG4gICAgICAgICAgICAgICAgaW5mbz17Y29sdW1uLmluZm99XG4gICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICB7Y29sdW1uLmxhYmVsfVxuICAgICAgICAgICAgICA8L0hlYWRlckNlbGw+XG4gICAgICAgICAgICApKX1cbiAgICAgICAgICA8L0hlYWRlclJvdz5cbiAgICAgICAgPC9IZWFkZXI+XG4gICAgICAgIHtsb2FkaW5nID8gKFxuICAgICAgICAgIDxCb2R5PlxuICAgICAgICAgICAgPFNrZWxldG9uUm93c1xuICAgICAgICAgICAgICBzZWxlY3RhYmxlPXtzZWxlY3RhYmxlfVxuICAgICAgICAgICAgICByb3dzPXs1fVxuICAgICAgICAgICAgICBjb2xzPXtjb2x1bW5zLmxlbmd0aH1cbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9Cb2R5PlxuICAgICAgICApIDogKFxuICAgICAgICAgIGNoaWxkcmVuXG4gICAgICAgICl9XG4gICAgICA8L1N0eWxlZFRhYmxlPlxuICAgIDwvVGFibGVQcm92aWRlcj5cbiAgKSxcbilcblxuLyoqXG4gKiBUYWJsZSBpcyBhIGNvbXBvbmVudCB0aGF0IGRpc3BsYXlzIGRhdGEgaW4gYSB0YWJ1bGFyIGZvcm1hdC5cbiAqL1xuZXhwb3J0IGNvbnN0IFRhYmxlID0gT2JqZWN0LmFzc2lnbihCYXNlVGFibGUsIHtcbiAgQm9keSxcbiAgUm93LFxuICBDZWxsLFxuICB1c2VUYWJsZUNvbnRleHQsXG4gIFNlbGVjdEJhcixcbn0pXG4iXX0= */"));
|
|
37
37
|
const BaseTable = React.forwardRef(({
|
|
38
38
|
selectable = false,
|
|
39
39
|
children,
|
|
@@ -2,7 +2,7 @@ import type { ComponentProps, ReactNode } from 'react';
|
|
|
2
2
|
import { HeaderCell } from './HeaderCell';
|
|
3
3
|
import { SelectBar } from './SelectBar';
|
|
4
4
|
type ColumnProps = Pick<ComponentProps<typeof HeaderCell>, 'isOrdered' | 'onOrder' | 'orderDirection'> & {
|
|
5
|
-
label?:
|
|
5
|
+
label?: ReactNode;
|
|
6
6
|
width?: string;
|
|
7
7
|
minWidth?: string;
|
|
8
8
|
maxWidth?: string;
|
|
@@ -29,7 +29,7 @@ const StyledTable = /* @__PURE__ */ _styled("table", process.env.NODE_ENV === "p
|
|
|
29
29
|
${bordered ? `& tbody tr {
|
|
30
30
|
border-bottom: 1px solid ${theme.colors.neutral.borderWeak};
|
|
31
31
|
}` : ""}
|
|
32
|
-
`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFtQm9CIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcywgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBmb3J3YXJkUmVmIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBCb2R5IH0gZnJvbSAnLi9Cb2R5J1xuaW1wb3J0IHsgQ2VsbCB9IGZyb20gJy4vQ2VsbCdcbmltcG9ydCB7IEhlYWRlciB9IGZyb20gJy4vSGVhZGVyJ1xuaW1wb3J0IHsgSGVhZGVyQ2VsbCB9IGZyb20gJy4vSGVhZGVyQ2VsbCdcbmltcG9ydCB7IEhlYWRlclJvdyB9IGZyb20gJy4vSGVhZGVyUm93J1xuaW1wb3J0IHsgUm93IH0gZnJvbSAnLi9Sb3cnXG5pbXBvcnQgeyBTZWxlY3RCYXIgfSBmcm9tICcuL1NlbGVjdEJhcidcbmltcG9ydCB7IFNrZWxldG9uUm93cyB9IGZyb20gJy4vU2tlbGV0b25Sb3dzJ1xuaW1wb3J0IHsgVGFibGVQcm92aWRlciwgdXNlVGFibGVDb250ZXh0IH0gZnJvbSAnLi9UYWJsZUNvbnRleHQnXG5cbnR5cGUgU3R5bGVkVGFibGVQcm9wcyA9IHtcbiAgc3RyaXBwZWQ6IGJvb2xlYW5cbiAgYm9yZGVyZWQ6IGJvb2xlYW5cbn1cbmNvbnN0IFN0eWxlZFRhYmxlID0gc3R5bGVkKCd0YWJsZScsIHtcbiAgc2hvdWxkRm9yd2FyZFByb3A6IHByb3AgPT4gIVsnYm9yZGVyZWQnLCAnc3RyaXBwZWQnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPFN0eWxlZFRhYmxlUHJvcHM+YFxuICB3aWR0aDogMTAwJTtcbiAgYm94LXNpemluZzogY29udGVudC1ib3g7XG4gIGJvcmRlci1jb2xsYXBzZTogY29sbGFwc2U7XG5cbiAgJHsoeyB0aGVtZSwgc3RyaXBwZWQsIGJvcmRlcmVkIH0pID0+IGBcbiAgJHtcbiAgICBzdHJpcHBlZFxuICAgICAgPyBgJiB0Ym9keSB0cjpudGgtb2YtdHlwZShldmVuKSB7XG4gICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kV2Vha307XG4gIH1gXG4gICAgICA6ICcnXG4gIH1cblxuICAke1xuICAgIGJvcmRlcmVkXG4gICAgICA/IGAmIHRib2R5IHRyIHtcbiAgICBib3JkZXItYm90dG9tOiAxcHggc29saWQgJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5ib3JkZXJXZWFrfTtcbiAgfWBcbiAgICAgIDogJydcbiAgfVxuICBgfVxuYFxuXG50eXBlIENvbHVtblByb3BzID0gUGljazxcbiAgQ29tcG9uZW50UHJvcHM8dHlwZW9mIEhlYWRlckNlbGw+
|
|
32
|
+
`, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFtQm9CIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL1RhYmxlL2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBDb21wb25lbnRQcm9wcywgUmVhY3ROb2RlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBmb3J3YXJkUmVmIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBCb2R5IH0gZnJvbSAnLi9Cb2R5J1xuaW1wb3J0IHsgQ2VsbCB9IGZyb20gJy4vQ2VsbCdcbmltcG9ydCB7IEhlYWRlciB9IGZyb20gJy4vSGVhZGVyJ1xuaW1wb3J0IHsgSGVhZGVyQ2VsbCB9IGZyb20gJy4vSGVhZGVyQ2VsbCdcbmltcG9ydCB7IEhlYWRlclJvdyB9IGZyb20gJy4vSGVhZGVyUm93J1xuaW1wb3J0IHsgUm93IH0gZnJvbSAnLi9Sb3cnXG5pbXBvcnQgeyBTZWxlY3RCYXIgfSBmcm9tICcuL1NlbGVjdEJhcidcbmltcG9ydCB7IFNrZWxldG9uUm93cyB9IGZyb20gJy4vU2tlbGV0b25Sb3dzJ1xuaW1wb3J0IHsgVGFibGVQcm92aWRlciwgdXNlVGFibGVDb250ZXh0IH0gZnJvbSAnLi9UYWJsZUNvbnRleHQnXG5cbnR5cGUgU3R5bGVkVGFibGVQcm9wcyA9IHtcbiAgc3RyaXBwZWQ6IGJvb2xlYW5cbiAgYm9yZGVyZWQ6IGJvb2xlYW5cbn1cbmNvbnN0IFN0eWxlZFRhYmxlID0gc3R5bGVkKCd0YWJsZScsIHtcbiAgc2hvdWxkRm9yd2FyZFByb3A6IHByb3AgPT4gIVsnYm9yZGVyZWQnLCAnc3RyaXBwZWQnXS5pbmNsdWRlcyhwcm9wKSxcbn0pPFN0eWxlZFRhYmxlUHJvcHM+YFxuICB3aWR0aDogMTAwJTtcbiAgYm94LXNpemluZzogY29udGVudC1ib3g7XG4gIGJvcmRlci1jb2xsYXBzZTogY29sbGFwc2U7XG5cbiAgJHsoeyB0aGVtZSwgc3RyaXBwZWQsIGJvcmRlcmVkIH0pID0+IGBcbiAgJHtcbiAgICBzdHJpcHBlZFxuICAgICAgPyBgJiB0Ym9keSB0cjpudGgtb2YtdHlwZShldmVuKSB7XG4gICAgYmFja2dyb3VuZDogJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5iYWNrZ3JvdW5kV2Vha307XG4gIH1gXG4gICAgICA6ICcnXG4gIH1cblxuICAke1xuICAgIGJvcmRlcmVkXG4gICAgICA/IGAmIHRib2R5IHRyIHtcbiAgICBib3JkZXItYm90dG9tOiAxcHggc29saWQgJHt0aGVtZS5jb2xvcnMubmV1dHJhbC5ib3JkZXJXZWFrfTtcbiAgfWBcbiAgICAgIDogJydcbiAgfVxuICBgfVxuYFxuXG50eXBlIENvbHVtblByb3BzID0gUGljazxcbiAgQ29tcG9uZW50UHJvcHM8dHlwZW9mIEhlYWRlckNlbGw+LFxuICAnaXNPcmRlcmVkJyB8ICdvbk9yZGVyJyB8ICdvcmRlckRpcmVjdGlvbidcbj4gJiB7XG4gIGxhYmVsPzogUmVhY3ROb2RlXG4gIHdpZHRoPzogc3RyaW5nXG4gIG1pbldpZHRoPzogc3RyaW5nXG4gIG1heFdpZHRoPzogc3RyaW5nXG4gIGluZm8/OiBzdHJpbmdcbn1cblxudHlwZSBUYWJsZVByb3BzID0ge1xuICBzZWxlY3RhYmxlPzogYm9vbGVhblxuICBjb2x1bW5zOiBDb2x1bW5Qcm9wc1tdXG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbiAgLyoqXG4gICAqIFNldCBpdCB0byB0cnVlIGlmIHlvdSB3YW50IHRvIGRpc3BsYXkgYSBwbGFjZWhvbGRlciBkdXJpbmcgbG9hZGluZ1xuICAgKiAqL1xuICBsb2FkaW5nPzogYm9vbGVhblxuICBib3JkZXJlZD86IGJvb2xlYW5cbiAgc3RyaXBwZWQ/OiBib29sZWFuXG59XG5cbmV4cG9ydCBjb25zdCBCYXNlVGFibGUgPSBmb3J3YXJkUmVmPEhUTUxUYWJsZUVsZW1lbnQsIFRhYmxlUHJvcHM+KFxuICAoXG4gICAge1xuICAgICAgc2VsZWN0YWJsZSA9IGZhbHNlLFxuICAgICAgY2hpbGRyZW4sXG4gICAgICBjb2x1bW5zLFxuICAgICAgbG9hZGluZyxcbiAgICAgIGJvcmRlcmVkID0gZmFsc2UsXG4gICAgICBzdHJpcHBlZCA9IGZhbHNlLFxuICAgIH0sXG4gICAgcmVmLFxuICApID0+IChcbiAgICA8VGFibGVQcm92aWRlclxuICAgICAgc2VsZWN0YWJsZT17c2VsZWN0YWJsZX1cbiAgICAgIHN0cmlwcGVkPXtzdHJpcHBlZH1cbiAgICAgIGJvcmRlcmVkPXtib3JkZXJlZH1cbiAgICA+XG4gICAgICA8U3R5bGVkVGFibGUgcmVmPXtyZWZ9IHN0cmlwcGVkPXtzdHJpcHBlZH0gYm9yZGVyZWQ9e2JvcmRlcmVkfT5cbiAgICAgICAgPEhlYWRlcj5cbiAgICAgICAgICA8SGVhZGVyUm93IGhhc1NlbGVjdEFsbENvbHVtbj17c2VsZWN0YWJsZX0+XG4gICAgICAgICAgICB7Y29sdW1ucy5tYXAoKGNvbHVtbiwgaW5kZXgpID0+IChcbiAgICAgICAgICAgICAgPEhlYWRlckNlbGxcbiAgICAgICAgICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcmVhY3Qvbm8tYXJyYXktaW5kZXgta2V5XG4gICAgICAgICAgICAgICAga2V5PXtgaGVhZGVyLWNvbHVtbi0ke2luZGV4fWB9XG4gICAgICAgICAgICAgICAgaXNPcmRlcmVkPXtjb2x1bW4uaXNPcmRlcmVkfVxuICAgICAgICAgICAgICAgIG9yZGVyRGlyZWN0aW9uPXtjb2x1bW4ub3JkZXJEaXJlY3Rpb259XG4gICAgICAgICAgICAgICAgb25PcmRlcj17Y29sdW1uLm9uT3JkZXJ9XG4gICAgICAgICAgICAgICAgd2lkdGg9e2NvbHVtbi53aWR0aH1cbiAgICAgICAgICAgICAgICBtaW5XaWR0aD17Y29sdW1uLm1pbldpZHRofVxuICAgICAgICAgICAgICAgIG1heFdpZHRoPXtjb2x1bW4ubWF4V2lkdGh9XG4gICAgICAgICAgICAgICAgaW5mbz17Y29sdW1uLmluZm99XG4gICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICB7Y29sdW1uLmxhYmVsfVxuICAgICAgICAgICAgICA8L0hlYWRlckNlbGw+XG4gICAgICAgICAgICApKX1cbiAgICAgICAgICA8L0hlYWRlclJvdz5cbiAgICAgICAgPC9IZWFkZXI+XG4gICAgICAgIHtsb2FkaW5nID8gKFxuICAgICAgICAgIDxCb2R5PlxuICAgICAgICAgICAgPFNrZWxldG9uUm93c1xuICAgICAgICAgICAgICBzZWxlY3RhYmxlPXtzZWxlY3RhYmxlfVxuICAgICAgICAgICAgICByb3dzPXs1fVxuICAgICAgICAgICAgICBjb2xzPXtjb2x1bW5zLmxlbmd0aH1cbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9Cb2R5PlxuICAgICAgICApIDogKFxuICAgICAgICAgIGNoaWxkcmVuXG4gICAgICAgICl9XG4gICAgICA8L1N0eWxlZFRhYmxlPlxuICAgIDwvVGFibGVQcm92aWRlcj5cbiAgKSxcbilcblxuLyoqXG4gKiBUYWJsZSBpcyBhIGNvbXBvbmVudCB0aGF0IGRpc3BsYXlzIGRhdGEgaW4gYSB0YWJ1bGFyIGZvcm1hdC5cbiAqL1xuZXhwb3J0IGNvbnN0IFRhYmxlID0gT2JqZWN0LmFzc2lnbihCYXNlVGFibGUsIHtcbiAgQm9keSxcbiAgUm93LFxuICBDZWxsLFxuICB1c2VUYWJsZUNvbnRleHQsXG4gIFNlbGVjdEJhcixcbn0pXG4iXX0= */"));
|
|
33
33
|
const BaseTable = forwardRef(({
|
|
34
34
|
selectable = false,
|
|
35
35
|
children,
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ultraviolet/ui",
|
|
3
|
-
"version": "1.67.
|
|
3
|
+
"version": "1.67.2",
|
|
4
4
|
"description": "Ultraviolet UI",
|
|
5
5
|
"homepage": "https://github.com/scaleway/ultraviolet#readme",
|
|
6
6
|
"repository": {
|
|
@@ -60,7 +60,7 @@
|
|
|
60
60
|
"react-dom": "18.3.1"
|
|
61
61
|
},
|
|
62
62
|
"devDependencies": {
|
|
63
|
-
"@babel/core": "7.
|
|
63
|
+
"@babel/core": "7.25.2",
|
|
64
64
|
"@emotion/react": "11.13.0",
|
|
65
65
|
"@emotion/styled": "11.13.0",
|
|
66
66
|
"@types/react": "18.3.3",
|
|
@@ -68,7 +68,7 @@
|
|
|
68
68
|
"@types/react-dom": "18.3.0",
|
|
69
69
|
"react": "18.3.1",
|
|
70
70
|
"react-dom": "18.3.1",
|
|
71
|
-
"@ultraviolet/icons": "2.16.
|
|
71
|
+
"@ultraviolet/icons": "2.16.2",
|
|
72
72
|
"@ultraviolet/themes": "1.12.4",
|
|
73
73
|
"@utils/test": "0.0.1"
|
|
74
74
|
},
|
|
@@ -88,7 +88,7 @@
|
|
|
88
88
|
"react-use-clipboard": "1.0.9",
|
|
89
89
|
"reakit": "1.3.11",
|
|
90
90
|
"@ultraviolet/themes": "1.12.4",
|
|
91
|
-
"@ultraviolet/icons": "2.16.
|
|
91
|
+
"@ultraviolet/icons": "2.16.2"
|
|
92
92
|
},
|
|
93
93
|
"scripts": {
|
|
94
94
|
"prebuild": "shx rm -rf dist",
|