@ultraviolet/ui 2.0.3 → 2.0.5
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.
- package/dist/components/List/Row.cjs +10 -8
- package/dist/components/List/Row.d.ts +3 -1
- package/dist/components/List/Row.js +10 -8
- package/dist/components/List/index.d.ts +2 -0
- package/dist/components/Menu/MenuContent.cjs +26 -14
- package/dist/components/Menu/MenuContent.d.ts +1 -0
- package/dist/components/Menu/MenuContent.js +26 -14
- package/dist/components/Menu/index.d.ts +1 -0
- package/dist/components/Menu/types.d.ts +4 -0
- package/dist/components/SelectInput/Dropdown.cjs +10 -10
- package/dist/components/SelectInput/Dropdown.js +10 -10
- package/dist/components/SelectInput/SelectBar.cjs +18 -12
- package/dist/components/SelectInput/SelectBar.js +19 -13
- package/dist/components/Table/HeaderCell.cjs +7 -6
- package/dist/components/Table/HeaderCell.d.ts +2 -1
- package/dist/components/Table/HeaderCell.js +7 -6
- package/dist/components/Table/HeaderRow.cjs +1 -1
- package/dist/components/Table/HeaderRow.js +1 -1
- package/package.json +2 -2
|
@@ -11,6 +11,7 @@ import { useMenu, DisclosureContext } from "./MenuProvider.js";
|
|
|
11
11
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() {
|
|
12
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
13
|
}
|
|
14
|
+
const SPACE_DISCLOSURE_POPUP = 24;
|
|
14
15
|
const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "production" ? {
|
|
15
16
|
shouldForwardProp: (prop) => !["searchable"].includes(prop),
|
|
16
17
|
target: "exosi9s4"
|
|
@@ -28,7 +29,7 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
|
|
|
28
29
|
searchable
|
|
29
30
|
}) => searchable ? `min-width: 20rem` : null, ";padding:", ({
|
|
30
31
|
theme
|
|
31
|
-
}) => `${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":"AAgC2B","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 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'].includes(prop),\n})<{ height: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: calc(${({ height }) => height} - ${({ theme }) =>\n    theme.space['0.5']});\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    }: 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 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    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            onKeyDown={handleKeyDown}\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"]} */"));
|
|
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"]} */"));
|
|
32
33
|
const Content = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
33
34
|
target: "exosi9s3"
|
|
34
35
|
} : {
|
|
@@ -39,7 +40,7 @@ const Content = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "product
|
|
|
39
40
|
styles: "overflow:auto"
|
|
40
41
|
} : {
|
|
41
42
|
name: "1qmr6ab",
|
|
42
|
-
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":"AAuD6B","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 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'].includes(prop),\n})<{ height: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: calc(${({ height }) => height} - ${({ theme }) =>\n    theme.space['0.5']});\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    }: 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 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    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            onKeyDown={handleKeyDown}\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"]} */",
|
|
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"]} */",
|
|
43
44
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
44
45
|
});
|
|
45
46
|
const Footer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
@@ -49,25 +50,25 @@ const Footer = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "producti
|
|
|
49
50
|
label: "Footer"
|
|
50
51
|
})("padding:", ({
|
|
51
52
|
theme
|
|
52
|
-
}) => 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":"AA2D4B","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 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'].includes(prop),\n})<{ height: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: calc(${({ height }) => height} - ${({ theme }) =>\n    theme.space['0.5']});\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    }: 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 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    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            onKeyDown={handleKeyDown}\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"]} */"));
|
|
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"]} */"));
|
|
53
54
|
const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
|
|
54
|
-
shouldForwardProp: (prop) => !["height"].includes(prop),
|
|
55
|
+
shouldForwardProp: (prop) => !["height", "heightAvailableSpace"].includes(prop),
|
|
55
56
|
target: "exosi9s1"
|
|
56
57
|
} : {
|
|
57
|
-
shouldForwardProp: (prop) => !["height"].includes(prop),
|
|
58
|
+
shouldForwardProp: (prop) => !["height", "heightAvailableSpace"].includes(prop),
|
|
58
59
|
target: "exosi9s1",
|
|
59
60
|
label: "MenuList"
|
|
60
|
-
})("overflow-y:auto;overflow-x:hidden;max-height:
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
}) => theme.space["0.5"]
|
|
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:", ({
|
|
65
66
|
theme
|
|
66
67
|
}) => theme.colors.other.elevation.background.raised, ";color:", ({
|
|
67
68
|
theme
|
|
68
69
|
}) => theme.colors.neutral.text, ";border-radius:", ({
|
|
69
70
|
theme
|
|
70
|
-
}) => 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":"AAiEsB","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 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'].includes(prop),\n})<{ height: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: calc(${({ height }) => height} - ${({ theme }) =>\n    theme.space['0.5']});\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    }: 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 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    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            onKeyDown={handleKeyDown}\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"]} */"));
|
|
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"]} */"));
|
|
71
72
|
const StyledSearchInput = /* @__PURE__ */ _styled(SearchInput, process.env.NODE_ENV === "production" ? {
|
|
72
73
|
target: "exosi9s0"
|
|
73
74
|
} : {
|
|
@@ -75,7 +76,7 @@ const StyledSearchInput = /* @__PURE__ */ _styled(SearchInput, process.env.NODE_
|
|
|
75
76
|
label: "StyledSearchInput"
|
|
76
77
|
})("padding:", ({
|
|
77
78
|
theme
|
|
78
|
-
}) => 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":"AA8F6C","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 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'].includes(prop),\n})<{ height: string }>`\n  overflow-y: auto;\n  overflow-x: hidden;\n  max-height: calc(${({ height }) => height} - ${({ theme }) =>\n    theme.space['0.5']});\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    }: 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 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    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            onKeyDown={handleKeyDown}\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"]} */"));
|
|
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"]} */"));
|
|
79
80
|
const Menu = forwardRef(({
|
|
80
81
|
id,
|
|
81
82
|
ariaLabel = "Menu",
|
|
@@ -91,7 +92,8 @@ const Menu = forwardRef(({
|
|
|
91
92
|
dynamicDomRendering,
|
|
92
93
|
align,
|
|
93
94
|
searchable = false,
|
|
94
|
-
footer
|
|
95
|
+
footer,
|
|
96
|
+
noShrink = false
|
|
95
97
|
}, ref) => {
|
|
96
98
|
const {
|
|
97
99
|
isVisible,
|
|
@@ -104,6 +106,7 @@ const Menu = forwardRef(({
|
|
|
104
106
|
} = useMenu();
|
|
105
107
|
const searchInputRef = useRef(null);
|
|
106
108
|
const [localChild, setLocalChild] = useState(null);
|
|
109
|
+
const [popupMaxHeight, setPopupMaxHeight] = useState(maxHeight ?? "30rem");
|
|
107
110
|
const contentRef = useRef(null);
|
|
108
111
|
const tempId = useId();
|
|
109
112
|
const finalId = `menu-${id ?? tempId}`;
|
|
@@ -206,6 +209,15 @@ const Menu = forwardRef(({
|
|
|
206
209
|
}
|
|
207
210
|
}
|
|
208
211
|
};
|
|
212
|
+
useEffect(() => {
|
|
213
|
+
if (disclosureRef.current && placement === "bottom" && !noShrink) {
|
|
214
|
+
const disclosureRect = disclosureRef.current.getBoundingClientRect();
|
|
215
|
+
const disclosureBottom = disclosureRect.bottom;
|
|
216
|
+
const targetSize = portalTarget.getBoundingClientRect().bottom;
|
|
217
|
+
const availableSpace = targetSize - disclosureBottom - SPACE_DISCLOSURE_POPUP;
|
|
218
|
+
setPopupMaxHeight(`${availableSpace}px`);
|
|
219
|
+
}
|
|
220
|
+
}, [isVisible, portalTarget, disclosureRef, placement, noShrink]);
|
|
209
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: () => {
|
|
210
222
|
setIsVisible(false);
|
|
211
223
|
setLocalChild(null);
|
|
@@ -213,7 +225,7 @@ const Menu = forwardRef(({
|
|
|
213
225
|
disclosureRef.current?.focus();
|
|
214
226
|
}
|
|
215
227
|
setShouldBeVisible(void 0);
|
|
216
|
-
}, 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", onKeyDown: handleKeyDown, role: "menu", children: [
|
|
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: [
|
|
217
229
|
/* @__PURE__ */ jsxs(Content, { ref: contentRef, children: [
|
|
218
230
|
searchable && typeof children !== "function" ? /* @__PURE__ */ jsx(StyledSearchInput, { onSearch, ref: searchInputRef, size: "small" }) : null,
|
|
219
231
|
finalChild
|
|
@@ -19,6 +19,7 @@ export declare const Menu: import("react").ForwardRefExoticComponent<{
|
|
|
19
19
|
hideOnClickItem?: boolean;
|
|
20
20
|
footer?: import("react").ReactNode;
|
|
21
21
|
placement?: Exclude<import("react").ComponentProps<typeof import("..").Popup>["placement"], "nested-menu">;
|
|
22
|
+
noShrink?: boolean;
|
|
22
23
|
} & Pick<{
|
|
23
24
|
id?: string;
|
|
24
25
|
children: import("react").ReactNode | ((renderProps: {
|
|
@@ -40,5 +40,9 @@ export type MenuProps = {
|
|
|
40
40
|
hideOnClickItem?: boolean;
|
|
41
41
|
footer?: ReactNode;
|
|
42
42
|
placement?: Exclude<ComponentProps<typeof Popup>['placement'], 'nested-menu'>;
|
|
43
|
+
/**
|
|
44
|
+
* When set to true, the menu does not shrink (height) to avoid overflow on the page
|
|
45
|
+
*/
|
|
46
|
+
noShrink?: boolean;
|
|
43
47
|
} & Pick<ComponentProps<typeof Popup>, 'dynamicDomRendering' | 'align'>;
|
|
44
48
|
export {};
|