@ultraviolet/ui 3.0.0-beta.21 → 3.0.0-beta.22

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.
Files changed (70) hide show
  1. package/dist/components/Avatar/styles.css.cjs +1 -0
  2. package/dist/components/Avatar/styles.css.js +1 -0
  3. package/dist/components/Avatar/variables.css.cjs +0 -1
  4. package/dist/components/Avatar/variables.css.js +0 -1
  5. package/dist/components/Breadcrumbs/components/Item.cjs +1 -1
  6. package/dist/components/Breadcrumbs/components/Item.js +2 -2
  7. package/dist/components/Breadcrumbs/components/styles.css.cjs +3 -1
  8. package/dist/components/Breadcrumbs/components/styles.css.d.ts +1 -0
  9. package/dist/components/Breadcrumbs/components/styles.css.js +3 -1
  10. package/dist/components/Button/index.cjs +2 -1
  11. package/dist/components/Button/index.d.ts +1 -0
  12. package/dist/components/Button/index.js +2 -1
  13. package/dist/components/Dialog/index.cjs +3 -21
  14. package/dist/components/Dialog/index.d.ts +1 -0
  15. package/dist/components/Dialog/index.js +3 -19
  16. package/dist/components/Dialog/styles.css.cjs +7 -0
  17. package/dist/components/Dialog/styles.css.d.ts +2 -0
  18. package/dist/components/Dialog/styles.css.js +7 -0
  19. package/dist/components/Link/index.cjs +2 -1
  20. package/dist/components/Link/index.d.ts +1 -0
  21. package/dist/components/Link/index.js +2 -1
  22. package/dist/components/List/Cell.cjs +21 -15
  23. package/dist/components/List/Cell.d.ts +2 -1
  24. package/dist/components/List/Cell.js +21 -13
  25. package/dist/components/List/ColumnProvider.cjs +28 -0
  26. package/dist/components/List/ColumnProvider.d.ts +14 -0
  27. package/dist/components/List/ColumnProvider.js +28 -0
  28. package/dist/components/List/HeaderCell.cjs +11 -45
  29. package/dist/components/List/HeaderCell.d.ts +2 -2
  30. package/dist/components/List/HeaderCell.js +12 -44
  31. package/dist/components/List/HeaderRow.cjs +4 -25
  32. package/dist/components/List/HeaderRow.js +4 -23
  33. package/dist/components/List/Row.cjs +23 -183
  34. package/dist/components/List/Row.d.ts +0 -16
  35. package/dist/components/List/Row.js +22 -180
  36. package/dist/components/List/SkeletonRows.cjs +7 -36
  37. package/dist/components/List/SkeletonRows.js +6 -33
  38. package/dist/components/List/index.cjs +4 -32
  39. package/dist/components/List/index.d.ts +1 -0
  40. package/dist/components/List/index.js +4 -30
  41. package/dist/components/List/styles.css.cjs +35 -0
  42. package/dist/components/List/styles.css.d.ts +24 -0
  43. package/dist/components/List/styles.css.js +35 -0
  44. package/dist/components/List/variables.css.cjs +16 -0
  45. package/dist/components/List/variables.css.d.ts +7 -0
  46. package/dist/components/List/variables.css.js +16 -0
  47. package/dist/components/Menu/MenuContent.cjs +13 -77
  48. package/dist/components/Menu/MenuContent.js +13 -75
  49. package/dist/components/Menu/components/Group.cjs +2 -12
  50. package/dist/components/Menu/components/Group.js +2 -10
  51. package/dist/components/Menu/components/Item.cjs +16 -94
  52. package/dist/components/Menu/components/Item.js +16 -92
  53. package/dist/components/Menu/styles.css.cjs +24 -0
  54. package/dist/components/Menu/styles.css.d.ts +47 -0
  55. package/dist/components/Menu/styles.css.js +24 -0
  56. package/dist/components/Row/index.cjs +15 -11
  57. package/dist/components/Row/index.d.ts +2 -1
  58. package/dist/components/Row/index.js +15 -11
  59. package/dist/components/Text/style.css.cjs +1 -0
  60. package/dist/components/Text/style.css.js +1 -0
  61. package/dist/components/Text/variables.css.cjs +0 -1
  62. package/dist/components/Text/variables.css.js +0 -1
  63. package/dist/components/UnitInput/styles.css.cjs +1 -0
  64. package/dist/components/UnitInput/styles.css.js +1 -0
  65. package/dist/ui.css +1 -1
  66. package/package.json +6 -6
  67. package/dist/components/List/constants.cjs +0 -5
  68. package/dist/components/List/constants.js +0 -5
  69. package/dist/components/Menu/constants.cjs +0 -7
  70. package/dist/components/Menu/constants.js +0 -7
@@ -2,85 +2,15 @@
2
2
  "use strict";
3
3
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
4
4
  const jsxRuntime = require("@emotion/react/jsx-runtime");
5
- const _styled = require("@emotion/styled/base");
5
+ const dynamic = require("@vanilla-extract/dynamic");
6
6
  const react = require("react");
7
7
  const index = require("../Popup/index.cjs");
8
8
  const index$2 = require("../SearchInput/index.cjs");
9
9
  const index$1 = require("../Stack/index.cjs");
10
- const constants = require("./constants.cjs");
11
10
  const helpers = require("./helpers.cjs");
12
11
  const MenuProvider = require("./MenuProvider.cjs");
13
- const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
14
- const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
15
- function _EMOTION_STRINGIFIED_CSS_ERROR__() {
16
- return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
17
- }
12
+ const styles_css = require("./styles.css.cjs");
18
13
  const SPACE_DISCLOSURE_POPUP = 24;
19
- const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, process.env.NODE_ENV === "production" ? {
20
- shouldForwardProp: (prop) => !["searchable"].includes(prop),
21
- target: "exosi9s4"
22
- } : {
23
- shouldForwardProp: (prop) => !["searchable"].includes(prop),
24
- target: "exosi9s4",
25
- label: "StyledPopup"
26
- })("background-color:", ({
27
- theme
28
- }) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
29
- theme
30
- }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
31
- theme
32
- }) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}min-width:", constants.SIZES.small, ";max-width:", constants.SIZES.large, ";", ({
33
- searchable
34
- }) => searchable ? `min-width: 20rem` : null, ";padding:", ({
35
- theme
36
- }) => `${theme.space["0.25"]} 0`, ";" + (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/Menu/MenuContent.tsx"],"names":[],"mappings":"AAkC2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
37
- const Content = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
38
- target: "exosi9s3"
39
- } : {
40
- target: "exosi9s3",
41
- label: "Content"
42
- })(process.env.NODE_ENV === "production" ? {
43
- name: "1qmr6ab",
44
- styles: "overflow:auto"
45
- } : {
46
- name: "1qmr6ab",
47
- styles: "overflow:auto/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx"],"names":[],"mappings":"AAyD6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */",
48
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
49
- });
50
- const Footer = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
51
- target: "exosi9s2"
52
- } : {
53
- target: "exosi9s2",
54
- label: "Footer"
55
- })("padding:", ({
56
- theme
57
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx"],"names":[],"mappings":"AA6D4B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
58
- const MenuList = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
59
- shouldForwardProp: (prop) => !["height", "heightAvailableSpace"].includes(prop),
60
- target: "exosi9s1"
61
- } : {
62
- shouldForwardProp: (prop) => !["height", "heightAvailableSpace"].includes(prop),
63
- target: "exosi9s1",
64
- label: "MenuList"
65
- })("overflow-y:auto;overflow-x:hidden;max-height:", ({
66
- theme,
67
- height,
68
- heightAvailableSpace
69
- }) => `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space["0.5"]})`, ";&:after,&:before{border:solid transparent;border-width:9px;content:' ';height:0;width:0;position:absolute;pointer-events:none;}&:after{border-color:transparent;}&:before{border-color:transparent;}background-color:", ({
70
- theme
71
- }) => theme.colors.other.elevation.background.raised, ";color:", ({
72
- theme
73
- }) => theme.colors.neutral.text, ";border-radius:", ({
74
- theme
75
- }) => theme.radii.default, ";position:relative;" + (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/Menu/MenuContent.tsx"],"names":[],"mappings":"AAmEoD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
76
- const StyledSearchInput = /* @__PURE__ */ _styled__default.default(index$2.SearchInput, process.env.NODE_ENV === "production" ? {
77
- target: "exosi9s0"
78
- } : {
79
- target: "exosi9s0",
80
- label: "StyledSearchInput"
81
- })("padding:", ({
82
- theme
83
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx"],"names":[],"mappings":"AAiG6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
84
14
  const Menu = react.forwardRef(({
85
15
  id,
86
16
  ariaLabel = "Menu",
@@ -222,19 +152,25 @@ const Menu = react.forwardRef(({
222
152
  setPopupMaxHeight(`${availableSpace}px`);
223
153
  }
224
154
  }, [isVisible, portalTarget, disclosureRef, placement, noShrink]);
225
- return /* @__PURE__ */ jsxRuntime.jsx(StyledPopup, { align, "aria-label": ariaLabel, className, "data-has-arrow": hasArrow, debounceDelay: triggerMethod === "hover" ? 250 : 0, dynamicDomRendering, hasArrow, hideOnClickOutside: true, id: finalId, maxHeight: maxHeight ?? "30rem", onClose: () => {
155
+ return /* @__PURE__ */ jsxRuntime.jsx(index.Popup, { align, "aria-label": ariaLabel, className: `${className ? `${className} ` : ""}${styles_css.menu({
156
+ arrow: hasArrow,
157
+ searchable
158
+ })}`, debounceDelay: triggerMethod === "hover" ? 250 : 0, dynamicDomRendering, hasArrow, hideOnClickOutside: true, id: finalId, maxHeight: maxHeight ?? "fit-content", onClose: () => {
226
159
  setIsVisible(false);
227
160
  setLocalChild(null);
228
161
  if (triggerMethod === "click") {
229
162
  disclosureRef.current?.focus();
230
163
  }
231
164
  setShouldBeVisible(void 0);
232
- }, onKeyDown: handleTabOpen, placement: isNested ? "nested-menu" : placement, portalTarget, ref: menuRef, role: "dialog", searchable, tabIndex: -1, text: /* @__PURE__ */ jsxRuntime.jsxs(MenuList, { className, "data-testid": dataTestId, height: maxHeight ?? "30rem", heightAvailableSpace: popupMaxHeight, onKeyDown: handleKeyDown, onMouseEnter: () => setShouldBeVisible(true), onMouseLeave: () => setShouldBeVisible(false), role: "menu", children: [
233
- /* @__PURE__ */ jsxRuntime.jsxs(Content, { ref: contentRef, children: [
234
- searchable && typeof children !== "function" ? /* @__PURE__ */ jsxRuntime.jsx(StyledSearchInput, { onSearch, ref: searchInputRef, size: "small" }) : null,
165
+ }, onKeyDown: handleTabOpen, placement: isNested ? "nested-menu" : placement, portalTarget, ref: menuRef, role: "dialog", tabIndex: -1, text: /* @__PURE__ */ jsxRuntime.jsxs(index$1.Stack, { className: `${className ? `${className} ` : ""}${styles_css.menuList}`, "data-testid": dataTestId, onKeyDown: handleKeyDown, onMouseEnter: () => setShouldBeVisible(true), onMouseLeave: () => setShouldBeVisible(false), role: "menu", style: dynamic.assignInlineVars({
166
+ [styles_css.heightMenu]: maxHeight ?? "30rem",
167
+ [styles_css.heightAvailableSpace]: popupMaxHeight
168
+ }), children: [
169
+ /* @__PURE__ */ jsxRuntime.jsxs(index$1.Stack, { className: styles_css.menuContent, ref: contentRef, children: [
170
+ searchable && typeof children !== "function" ? /* @__PURE__ */ jsxRuntime.jsx(index$2.SearchInput, { className: styles_css.menuSearchInput, onSearch, ref: searchInputRef, size: "small" }) : null,
235
171
  finalChild
236
172
  ] }),
237
- footer ? /* @__PURE__ */ jsxRuntime.jsx(Footer, { children: footer }) : null
173
+ footer ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Stack, { className: styles_css.menuFooter, children: footer }) : null
238
174
  ] }), visible: triggerMethod === "click" ? isVisible : shouldBeVisible, children: /* @__PURE__ */ jsxRuntime.jsx(MenuProvider.DisclosureContext.Provider, { value: true, children: finalDisclosure }) });
239
175
  });
240
176
  exports.Menu = Menu;
@@ -1,82 +1,14 @@
1
1
  "use client";
2
2
  import { jsx, jsxs } from "@emotion/react/jsx-runtime";
3
- import _styled from "@emotion/styled/base";
3
+ import { assignInlineVars } from "@vanilla-extract/dynamic";
4
4
  import { forwardRef, useRef, useState, useId, isValidElement, useImperativeHandle, cloneElement, useCallback, useEffect, useMemo } from "react";
5
5
  import { Popup } from "../Popup/index.js";
6
6
  import { SearchInput } from "../SearchInput/index.js";
7
7
  import { Stack } from "../Stack/index.js";
8
- import { SIZES } from "./constants.js";
9
8
  import { searchChildren, getListItem } from "./helpers.js";
10
9
  import { useMenu, DisclosureContext } from "./MenuProvider.js";
11
- function _EMOTION_STRINGIFIED_CSS_ERROR__() {
12
- return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
13
- }
10
+ import { menu, menuList, menuContent, menuSearchInput, menuFooter, heightAvailableSpace, heightMenu } from "./styles.css.js";
14
11
  const SPACE_DISCLOSURE_POPUP = 24;
15
- const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "production" ? {
16
- shouldForwardProp: (prop) => !["searchable"].includes(prop),
17
- target: "exosi9s4"
18
- } : {
19
- shouldForwardProp: (prop) => !["searchable"].includes(prop),
20
- target: "exosi9s4",
21
- label: "StyledPopup"
22
- })("background-color:", ({
23
- theme
24
- }) => theme.colors.other.elevation.background.raised, ";box-shadow:", ({
25
- theme
26
- }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`, ";padding:0;&[data-has-arrow='true']{&::after{border-color:", ({
27
- theme
28
- }) => theme.colors.other.elevation.background.raised, " transparent transparent transparent;}}min-width:", SIZES.small, ";max-width:", SIZES.large, ";", ({
29
- searchable
30
- }) => searchable ? `min-width: 20rem` : null, ";padding:", ({
31
- theme
32
- }) => `${theme.space["0.25"]} 0`, ";" + (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/Menu/MenuContent.tsx"],"names":[],"mappings":"AAkC2B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
33
- const Content = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
34
- target: "exosi9s3"
35
- } : {
36
- target: "exosi9s3",
37
- label: "Content"
38
- })(process.env.NODE_ENV === "production" ? {
39
- name: "1qmr6ab",
40
- styles: "overflow:auto"
41
- } : {
42
- name: "1qmr6ab",
43
- styles: "overflow:auto/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx"],"names":[],"mappings":"AAyD6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */",
44
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
45
- });
46
- const Footer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
47
- target: "exosi9s2"
48
- } : {
49
- target: "exosi9s2",
50
- label: "Footer"
51
- })("padding:", ({
52
- theme
53
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx"],"names":[],"mappings":"AA6D4B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
54
- const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
55
- shouldForwardProp: (prop) => !["height", "heightAvailableSpace"].includes(prop),
56
- target: "exosi9s1"
57
- } : {
58
- shouldForwardProp: (prop) => !["height", "heightAvailableSpace"].includes(prop),
59
- target: "exosi9s1",
60
- label: "MenuList"
61
- })("overflow-y:auto;overflow-x:hidden;max-height:", ({
62
- theme,
63
- height,
64
- heightAvailableSpace
65
- }) => `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space["0.5"]})`, ";&:after,&:before{border:solid transparent;border-width:9px;content:' ';height:0;width:0;position:absolute;pointer-events:none;}&:after{border-color:transparent;}&:before{border-color:transparent;}background-color:", ({
66
- theme
67
- }) => theme.colors.other.elevation.background.raised, ";color:", ({
68
- theme
69
- }) => theme.colors.neutral.text, ";border-radius:", ({
70
- theme
71
- }) => theme.radii.default, ";position:relative;" + (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/Menu/MenuContent.tsx"],"names":[],"mappings":"AAmEoD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
72
- const StyledSearchInput = /* @__PURE__ */ _styled(SearchInput, process.env.NODE_ENV === "production" ? {
73
- target: "exosi9s0"
74
- } : {
75
- target: "exosi9s0",
76
- label: "StyledSearchInput"
77
- })("padding:", ({
78
- theme
79
- }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx"],"names":[],"mappings":"AAiG6C","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/Menu/MenuContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  KeyboardEvent,\n  MouseEvent,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useCallback,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { SearchInput } from '../SearchInput'\nimport { Stack } from '../Stack'\nimport { SIZES } from './constants'\nimport { getListItem, searchChildren } from './helpers'\nimport { DisclosureContext, useMenu } from './MenuProvider'\nimport type { MenuProps } from './types'\n\nconst SPACE_DISCLOSURE_POPUP = 24 // in px\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['searchable'].includes(prop),\n})<{ searchable: boolean }>`\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) =>\n    `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) =>\n        theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  min-width: ${SIZES.small};\n  max-width: ${SIZES.large};\n\n  ${({ searchable }) => (searchable ? `min-width: 20rem` : null)};\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n\n`\n\nconst Content = styled(Stack)`\noverflow: auto;\n`\n\nconst Footer = styled(Stack)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nconst MenuList = styled(Stack, {\n  shouldForwardProp: prop => !['height', 'heightAvailableSpace'].includes(prop),\n})<{ height: string; heightAvailableSpace: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: ${({ theme, height, heightAvailableSpace }) =>\n    `calc(min(${height}, ${heightAvailableSpace}) - ${theme.space['0.5']})`};\n\n  &:after,\n  &:before {\n    border: solid transparent;\n    border-width: 9px;\n    content: ' ';\n    height: 0;\n    width: 0;\n    position: absolute;\n    pointer-events: none;\n  }\n\n  &:after {\n    border-color: transparent;\n  }\n  &:before {\n    border-color: transparent;\n  }\n  background-color: ${({ theme }) =>\n    theme.colors.other.elevation.background.raised};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  border-radius: ${({ theme }) => theme.radii.default};\n  position: relative;\n`\n\nconst StyledSearchInput = styled(SearchInput)`\n  padding: ${({ theme }) => theme.space['1']};\n`\n\nexport const Menu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      portalTarget = document.body,\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n      searchable = false,\n      footer,\n      noShrink = false,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const {\n      isVisible,\n      setIsVisible,\n      isNested,\n      disclosureRef,\n      menuRef,\n      setShouldBeVisible,\n      shouldBeVisible,\n    } = useMenu()\n    const searchInputRef = useRef<HTMLInputElement>(null)\n    const [localChild, setLocalChild] = useState<ReactNode[] | null>(null)\n    const [popupMaxHeight, setPopupMaxHeight] = useState<string>(\n      maxHeight ?? '30rem',\n    )\n    const contentRef = useRef<HTMLDivElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n    // if you need dialog inside your component, use function, otherwise component is fine\n    const target = isValidElement<ButtonHTMLAttributes<HTMLButtonElement>>(\n      disclosure,\n    )\n      ? disclosure\n      : disclosure({ visible: isVisible })\n    const innerRef = useRef(target as unknown as HTMLButtonElement)\n    useImperativeHandle(ref, () => innerRef.current)\n\n    const finalDisclosure = cloneElement(target, {\n      'aria-expanded': isVisible,\n      'aria-haspopup': 'dialog',\n      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    const onSearch = useCallback(\n      (value: string) => {\n        if (typeof children === 'object') {\n          setLocalChild(searchChildren(children, value))\n        }\n      },\n      [children],\n    )\n\n    useEffect(() => {\n      if (isVisible && searchable) {\n        setTimeout(() => {\n          searchInputRef.current?.focus()\n        }, 50)\n      }\n    }, [isVisible, searchable])\n\n    useEffect(() => {\n      if (disclosureRef.current && triggerMethod === 'hover') {\n        const handler = (value: boolean | undefined) => {\n          setShouldBeVisible(value)\n        }\n\n        disclosureRef.current.addEventListener('focus', () => handler(true))\n        disclosureRef.current.addEventListener('mouseenter', () =>\n          handler(true),\n        )\n        disclosureRef.current.addEventListener('mouseleave', () =>\n          handler(false),\n        )\n        disclosureRef.current.addEventListener('keydown', event => {\n          if (event.key === 'ArrowDown' || event.key === 'ArrowUp') {\n            handler(false) // force close menu when navigating with arrow keys\n          }\n        })\n\n        return () => {\n          window.removeEventListener('focus', () => handler(undefined))\n          window.removeEventListener('mouseenter', () => handler(undefined))\n          window.removeEventListener('mouseleave', () => handler(undefined))\n          window.removeEventListener('keydown', () => handler(undefined))\n        }\n      }\n\n      return undefined\n    }, [setShouldBeVisible, disclosureRef, triggerMethod])\n\n    const finalChild = useMemo(() => {\n      if (typeof children === 'function') {\n        return children({ toggle: () => setIsVisible(!isVisible) })\n      }\n\n      if (searchable && localChild) {\n        return localChild\n      }\n\n      return children\n    }, [children, isVisible, localChild, searchable, setIsVisible])\n\n    const handleTabOpen = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem && isVisible && ['Tab', 'ArrowDown'].includes(event.key)) {\n          event?.preventDefault()\n          listItem[0]?.focus()\n        }\n      }\n    }\n\n    const handleKeyDown = (event: KeyboardEvent) => {\n      if (contentRef.current) {\n        const listItem = getListItem([...contentRef.current.children])\n        if (listItem) {\n          const currentElement = listItem.find(\n            item => item === document.activeElement,\n          )\n          if (currentElement) {\n            if (event.key === 'ArrowDown') {\n              event.preventDefault()\n              const indexOfCurrent = listItem.indexOf(currentElement)\n\n              if (indexOfCurrent < listItem.length - 1) {\n                listItem[indexOfCurrent + 1].focus()\n              } else {\n                listItem[0].focus()\n              }\n            } else if (event.key === 'ArrowUp') {\n              event.preventDefault()\n\n              const indexOfCurrent = listItem.indexOf(currentElement)\n              if (indexOfCurrent > 0) {\n                listItem[indexOfCurrent - 1].focus()\n              } else {\n                listItem[listItem.length - 1].focus()\n              }\n            } else if (event.key === 'ArrowLeft' && triggerMethod === 'hover') {\n              disclosureRef.current?.focus()\n              setShouldBeVisible(undefined)\n            }\n          }\n        }\n      }\n    }\n\n    useEffect(() => {\n      if (disclosureRef.current && placement === 'bottom' && !noShrink) {\n        const disclosureRect = disclosureRef.current.getBoundingClientRect()\n        const disclosureBottom = disclosureRect.bottom\n        const targetSize = portalTarget.getBoundingClientRect().bottom\n        const availableSpace =\n          targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP\n        setPopupMaxHeight(`${availableSpace}px`)\n      }\n    }, [isVisible, portalTarget, disclosureRef, placement, noShrink])\n\n    return (\n      <StyledPopup\n        align={align}\n        aria-label={ariaLabel}\n        className={className}\n        data-has-arrow={hasArrow}\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        dynamicDomRendering={dynamicDomRendering}\n        hasArrow={hasArrow}\n        hideOnClickOutside\n        id={finalId}\n        maxHeight={maxHeight ?? '30rem'}\n        onClose={() => {\n          setIsVisible(false)\n          setLocalChild(null)\n          if (triggerMethod === 'click') {\n            disclosureRef.current?.focus()\n          }\n          setShouldBeVisible(undefined)\n        }}\n        onKeyDown={handleTabOpen}\n        placement={isNested ? 'nested-menu' : placement}\n        portalTarget={portalTarget}\n        ref={menuRef}\n        role=\"dialog\"\n        searchable={searchable}\n        tabIndex={-1}\n        text={\n          <MenuList\n            className={className}\n            data-testid={dataTestId}\n            height={maxHeight ?? '30rem'}\n            heightAvailableSpace={popupMaxHeight}\n            onKeyDown={handleKeyDown}\n            onMouseEnter={() => setShouldBeVisible(true)}\n            onMouseLeave={() => setShouldBeVisible(false)}\n            role=\"menu\"\n          >\n            <Content ref={contentRef}>\n              {searchable && typeof children !== 'function' ? (\n                <StyledSearchInput\n                  onSearch={onSearch}\n                  ref={searchInputRef}\n                  size=\"small\"\n                />\n              ) : null}\n              {finalChild}\n            </Content>\n            {footer ? <Footer>{footer}</Footer> : null}\n          </MenuList>\n        }\n        visible={triggerMethod === 'click' ? isVisible : shouldBeVisible}\n      >\n        <DisclosureContext.Provider value>\n          {finalDisclosure}\n        </DisclosureContext.Provider>\n      </StyledPopup>\n    )\n  },\n)\n"]} */"));
80
12
  const Menu = forwardRef(({
81
13
  id,
82
14
  ariaLabel = "Menu",
@@ -218,19 +150,25 @@ const Menu = forwardRef(({
218
150
  setPopupMaxHeight(`${availableSpace}px`);
219
151
  }
220
152
  }, [isVisible, portalTarget, disclosureRef, placement, noShrink]);
221
- return /* @__PURE__ */ jsx(StyledPopup, { align, "aria-label": ariaLabel, className, "data-has-arrow": hasArrow, debounceDelay: triggerMethod === "hover" ? 250 : 0, dynamicDomRendering, hasArrow, hideOnClickOutside: true, id: finalId, maxHeight: maxHeight ?? "30rem", onClose: () => {
153
+ return /* @__PURE__ */ jsx(Popup, { align, "aria-label": ariaLabel, className: `${className ? `${className} ` : ""}${menu({
154
+ arrow: hasArrow,
155
+ searchable
156
+ })}`, debounceDelay: triggerMethod === "hover" ? 250 : 0, dynamicDomRendering, hasArrow, hideOnClickOutside: true, id: finalId, maxHeight: maxHeight ?? "fit-content", onClose: () => {
222
157
  setIsVisible(false);
223
158
  setLocalChild(null);
224
159
  if (triggerMethod === "click") {
225
160
  disclosureRef.current?.focus();
226
161
  }
227
162
  setShouldBeVisible(void 0);
228
- }, onKeyDown: handleTabOpen, placement: isNested ? "nested-menu" : placement, portalTarget, ref: menuRef, role: "dialog", searchable, tabIndex: -1, text: /* @__PURE__ */ jsxs(MenuList, { className, "data-testid": dataTestId, height: maxHeight ?? "30rem", heightAvailableSpace: popupMaxHeight, onKeyDown: handleKeyDown, onMouseEnter: () => setShouldBeVisible(true), onMouseLeave: () => setShouldBeVisible(false), role: "menu", children: [
229
- /* @__PURE__ */ jsxs(Content, { ref: contentRef, children: [
230
- searchable && typeof children !== "function" ? /* @__PURE__ */ jsx(StyledSearchInput, { onSearch, ref: searchInputRef, size: "small" }) : null,
163
+ }, onKeyDown: handleTabOpen, placement: isNested ? "nested-menu" : placement, portalTarget, ref: menuRef, role: "dialog", tabIndex: -1, text: /* @__PURE__ */ jsxs(Stack, { className: `${className ? `${className} ` : ""}${menuList}`, "data-testid": dataTestId, onKeyDown: handleKeyDown, onMouseEnter: () => setShouldBeVisible(true), onMouseLeave: () => setShouldBeVisible(false), role: "menu", style: assignInlineVars({
164
+ [heightMenu]: maxHeight ?? "30rem",
165
+ [heightAvailableSpace]: popupMaxHeight
166
+ }), children: [
167
+ /* @__PURE__ */ jsxs(Stack, { className: menuContent, ref: contentRef, children: [
168
+ searchable && typeof children !== "function" ? /* @__PURE__ */ jsx(SearchInput, { className: menuSearchInput, onSearch, ref: searchInputRef, size: "small" }) : null,
231
169
  finalChild
232
170
  ] }),
233
- footer ? /* @__PURE__ */ jsx(Footer, { children: footer }) : null
171
+ footer ? /* @__PURE__ */ jsx(Stack, { className: menuFooter, children: footer }) : null
234
172
  ] }), visible: triggerMethod === "click" ? isVisible : shouldBeVisible, children: /* @__PURE__ */ jsx(DisclosureContext.Provider, { value: true, children: finalDisclosure }) });
235
173
  });
236
174
  export {
@@ -2,20 +2,10 @@
2
2
  "use strict";
3
3
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
4
4
  const jsxRuntime = require("@emotion/react/jsx-runtime");
5
- const _styled = require("@emotion/styled/base");
6
5
  const react = require("react");
7
6
  const index = require("../../Stack/index.cjs");
8
7
  const index$1 = require("../../Text/index.cjs");
9
- const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
10
- const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
11
- const Container = /* @__PURE__ */ _styled__default.default("span", process.env.NODE_ENV === "production" ? {
12
- target: "e13wft2v0"
13
- } : {
14
- target: "e13wft2v0",
15
- label: "Container"
16
- })("padding:", ({
17
- theme
18
- }) => `${theme.space["0.5"]} ${theme.space["1.5"]}`, ";text-align:left;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnUvY29tcG9uZW50cy9Hcm91cC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBUTZCIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL01lbnUvY29tcG9uZW50cy9Hcm91cC50c3giLCJzb3VyY2VzQ29udGVudCI6WyIndXNlIGNsaWVudCdcblxuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IFJlYWN0Tm9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQ2hpbGRyZW4gfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vLi4vVGV4dCdcblxuY29uc3QgQ29udGFpbmVyID0gc3R5bGVkLnNwYW5gXG4gIHBhZGRpbmc6ICR7KHsgdGhlbWUgfSkgPT4gYCR7dGhlbWUuc3BhY2VbJzAuNSddfSAke3RoZW1lLnNwYWNlWycxLjUnXX1gfTtcbiAgdGV4dC1hbGlnbjogbGVmdDtcbmBcblxudHlwZSBHcm91cFByb3BzID0ge1xuICBsYWJlbDogc3RyaW5nXG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbiAgbGFiZWxEZXNjcmlwdGlvbj86IFJlYWN0Tm9kZVxuICAvKipcbiAgICogRW1wdHkgc3RhdGUgd2lsbCBiZSBzaG93biB3aGVuIHRoZXJlIGFyZSBubyBjaGlsZHJlblxuICAgKi9cbiAgZW1wdHlTdGF0ZT86IFJlYWN0Tm9kZVxufVxuXG5leHBvcnQgY29uc3QgR3JvdXAgPSAoe1xuICBsYWJlbCxcbiAgY2hpbGRyZW4sXG4gIGxhYmVsRGVzY3JpcHRpb24sXG4gIGVtcHR5U3RhdGUsXG59OiBHcm91cFByb3BzKSA9PiB7XG4gIGNvbnN0IGlzQ2hpbGRyZW5FbXB0eSA9IENoaWxkcmVuLmNvdW50KGNoaWxkcmVuKSA9PT0gMFxuXG4gIHJldHVybiAoXG4gICAgPD5cbiAgICAgIDxDb250YWluZXI+XG4gICAgICAgIDxTdGFjayBhbGlnbkl0ZW1zPVwiY2VudGVyXCIgZGlyZWN0aW9uPVwicm93XCIgZ2FwPXsxfT5cbiAgICAgICAgICA8VGV4dFxuICAgICAgICAgICAgYXM9XCJzcGFuXCJcbiAgICAgICAgICAgIHByb21pbmVuY2U9XCJ3ZWFrXCJcbiAgICAgICAgICAgIHNlbnRpbWVudD1cIm5ldXRyYWxcIlxuICAgICAgICAgICAgdmFyaWFudD1cImNhcHRpb25TdHJvbmdcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIHtsYWJlbH1cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAge2xhYmVsRGVzY3JpcHRpb24gfHwgbnVsbH1cbiAgICAgICAgPC9TdGFjaz5cbiAgICAgIDwvQ29udGFpbmVyPlxuICAgICAge2lzQ2hpbGRyZW5FbXB0eSAmJiBlbXB0eVN0YXRlID8gZW1wdHlTdGF0ZSA6IGNoaWxkcmVufVxuICAgIDwvPlxuICApXG59XG4iXX0= */"));
8
+ const styles_css = require("../styles.css.cjs");
19
9
  const Group = ({
20
10
  label,
21
11
  children,
@@ -24,7 +14,7 @@ const Group = ({
24
14
  }) => {
25
15
  const isChildrenEmpty = react.Children.count(children) === 0;
26
16
  return /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
27
- /* @__PURE__ */ jsxRuntime.jsx(Container, { children: /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { alignItems: "center", direction: "row", gap: 1, children: [
17
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: styles_css.menuGroup, children: /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { alignItems: "center", direction: "row", gap: 1, children: [
28
18
  /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "span", prominence: "weak", sentiment: "neutral", variant: "captionStrong", children: label }),
29
19
  labelDescription || null
30
20
  ] }) }),