@ultraviolet/ui 1.81.1 → 1.81.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -31,7 +31,7 @@ const StyledPopup = /* @__PURE__ */ _styled__default.default(index.Popup, proces
31
31
  size
32
32
  }) => SIZES[size], ";max-width:none;padding:", ({
33
33
  theme
34
- }) => `${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/MenuV2/index.tsx"],"names":[],"mappings":"AAyCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
34
+ }) => `${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/MenuV2/index.tsx"],"names":[],"mappings":"AAyCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
35
35
  const MenuList = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
36
36
  target: "e1jn11gg0"
37
37
  } : {
@@ -43,7 +43,7 @@ const MenuList = /* @__PURE__ */ _styled__default.default(index$1.Stack, process
43
43
  theme
44
44
  }) => theme.colors.neutral.text, ";border-radius:", ({
45
45
  theme
46
- }) => 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/MenuV2/index.tsx"],"names":[],"mappings":"AA0D8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
46
+ }) => 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/MenuV2/index.tsx"],"names":[],"mappings":"AA0D8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
47
47
  const FwdMenu = React.forwardRef(({
48
48
  id,
49
49
  ariaLabel = "Menu",
@@ -56,7 +56,7 @@ const FwdMenu = React.forwardRef(({
56
56
  "data-testid": dataTestId,
57
57
  maxHeight,
58
58
  maxWidth,
59
- portalTarget,
59
+ portalTarget = document.body,
60
60
  size = "small",
61
61
  triggerMethod = "click",
62
62
  dynamicDomRendering,
@@ -27,7 +27,7 @@ const StyledPopup = /* @__PURE__ */ _styled(Popup, process.env.NODE_ENV === "pro
27
27
  size
28
28
  }) => SIZES[size], ";max-width:none;padding:", ({
29
29
  theme
30
- }) => `${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/MenuV2/index.tsx"],"names":[],"mappings":"AAyCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
30
+ }) => `${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/MenuV2/index.tsx"],"names":[],"mappings":"AAyCgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
31
31
  const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
32
32
  target: "e1jn11gg0"
33
33
  } : {
@@ -39,7 +39,7 @@ const MenuList = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produc
39
39
  theme
40
40
  }) => theme.colors.neutral.text, ";border-radius:", ({
41
41
  theme
42
- }) => 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/MenuV2/index.tsx"],"names":[],"mappings":"AA0D8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
42
+ }) => 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/MenuV2/index.tsx"],"names":[],"mappings":"AA0D8B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/MenuV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ButtonHTMLAttributes,\n  ComponentProps,\n  MouseEvent,\n  ReactElement,\n  ReactNode,\n  Ref,\n} from 'react'\nimport {\n  cloneElement,\n  forwardRef,\n  isValidElement,\n  useId,\n  useImperativeHandle,\n  useRef,\n  useState,\n} from 'react'\nimport { Popup } from '../Popup'\nimport { Stack } from '../Stack'\nimport { Group } from './Group'\nimport Item from './Item'\n\nconst SIZES = {\n  small: '180px',\n  medium: '280px',\n  large: '380px',\n}\n\nexport type DisclosureProps = { visible: boolean }\n\ntype DisclosureElement =\n  | ((\n      disclosure: DisclosureProps,\n    ) => ReactElement<ButtonHTMLAttributes<HTMLButtonElement>>)\n  | (ReactElement<ButtonHTMLAttributes<HTMLButtonElement>> & {\n      ref?: Ref<HTMLButtonElement>\n    })\n\nconst StyledPopup = styled(Popup, {\n  shouldForwardProp: prop => !['size'].includes(prop),\n})<{ size: keyof typeof SIZES }>`\n  background-color: ${({ theme }) => theme.colors.other.elevation.background.raised};\n  box-shadow: ${({ theme }) => `${theme.shadows.raised[0]}, ${theme.shadows.raised[1]}`};\n  padding: 0;\n\n  &[data-has-arrow='true'] {\n    &::after {\n      border-color: ${({ theme }) => theme.colors.other.elevation.background.raised}\n        transparent transparent transparent;\n    }\n  }\n\n  width: ${({ size }) => SIZES[size]};\n  max-width: none;\n  padding: ${({ theme }) => `${theme.space['0.25']} 0`};\n`\n\nconst MenuList = styled(Stack)`\n  overflow-y: auto;\n  overflow-x: hidden;\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\ntype ChildMenuProps = {\n  toggle: () => void\n}\n\ntype MenuProps = {\n  id?: string\n  ariaLabel?: string\n  children?: ReactNode | (({ toggle }: ChildMenuProps) => ReactNode)\n  className?: string\n  disclosure: DisclosureElement\n  hasArrow?: boolean\n  visible?: boolean\n  'data-testid'?: string\n  maxHeight?: string\n  /**\n   * @deprecated: use `size` instead\n   */\n  maxWidth?: string\n  /**\n   * By default, the portal target is children container or document.body if children is a function. You can override this\n   * behavior by setting a portalTarget prop.\n   */\n  portalTarget?: HTMLElement\n  size?: keyof typeof SIZES\n  /**\n   * The behavior of the menu when it is opened. If set to `click`, the menu will open when the user clicks on the disclosure.\n   * If set to `hover`, the menu will open when the user hovers over the disclosure.\n   */\n  triggerMethod?: 'click' | 'hover'\n} & Pick<\n  ComponentProps<typeof Popup>,\n  'placement' | 'dynamicDomRendering' | 'align'\n>\n\nconst FwdMenu = forwardRef(\n  (\n    {\n      id,\n      ariaLabel = 'Menu',\n      children,\n      disclosure,\n      hasArrow = false,\n      placement = 'bottom',\n      visible = false,\n      className,\n      'data-testid': dataTestId,\n      maxHeight,\n      maxWidth,\n      portalTarget = document.body,\n      size = 'small',\n      triggerMethod = 'click',\n      dynamicDomRendering,\n      align,\n    }: MenuProps,\n    ref: Ref<HTMLButtonElement | null>,\n  ) => {\n    const [isVisible, setIsVisible] = useState(visible)\n    const popupRef = useRef<HTMLDivElement>(null)\n    const disclosureRef = useRef<HTMLButtonElement>(null)\n    const tempId = useId()\n    const finalId = `menu-${id ?? tempId}`\n\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      onClick: (event: MouseEvent<HTMLButtonElement>) => {\n        target.props.onClick?.(event)\n        setIsVisible(!isVisible)\n      },\n      'aria-haspopup': 'dialog',\n      'aria-expanded': isVisible,\n      // @ts-expect-error not sure how to fix this\n      ref: disclosureRef,\n    })\n\n    return (\n      <StyledPopup\n        debounceDelay={triggerMethod === 'hover' ? 250 : 0}\n        hideOnClickOutside\n        aria-label={ariaLabel}\n        className={className}\n        visible={triggerMethod === 'click' ? isVisible : undefined}\n        placement={placement}\n        hasArrow={hasArrow}\n        data-has-arrow={hasArrow}\n        role=\"dialog\"\n        id={finalId}\n        ref={popupRef}\n        onClose={() => setIsVisible(false)}\n        tabIndex={-1}\n        maxHeight={maxHeight ?? '480px'}\n        maxWidth={maxWidth}\n        size={size}\n        text={\n          <MenuList data-testid={dataTestId} className={className} role=\"menu\">\n            {typeof children === 'function'\n              ? children({ toggle: () => setIsVisible(!isVisible) })\n              : children}\n          </MenuList>\n        }\n        portalTarget={portalTarget}\n        dynamicDomRendering={dynamicDomRendering}\n        align={align}\n      >\n        {finalDisclosure}\n      </StyledPopup>\n    )\n  },\n)\n\n/**\n * A menu is a widget that offers a list of choices to the user, such as a set of actions or functions.\n * A menu is usually opened, or made visible, by activating a menu button, choosing an item in a menu that opens a\n * sub menu, or by invoking a command, such as `Shift + F10` on Windows, that opens a context specific menu.\n * When a user activates a choice in a menu, the menu usually closes unless the choice opened a submenu.\n */\nexport const MenuV2 = Object.assign(FwdMenu, { Item, Group })\n"]} */"));
43
43
  const FwdMenu = forwardRef(({
44
44
  id,
45
45
  ariaLabel = "Menu",
@@ -52,7 +52,7 @@ const FwdMenu = forwardRef(({
52
52
  "data-testid": dataTestId,
53
53
  maxHeight,
54
54
  maxWidth,
55
- portalTarget,
55
+ portalTarget = document.body,
56
56
  size = "small",
57
57
  triggerMethod = "click",
58
58
  dynamicDomRendering,
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "1.81.1",
3
+ "version": "1.81.2",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {
@@ -63,9 +63,9 @@
63
63
  "@babel/core": "7.26.0",
64
64
  "@emotion/react": "11.14.0",
65
65
  "@emotion/styled": "11.14.0",
66
- "@types/react": "18.3.14",
66
+ "@types/react": "18.3.16",
67
67
  "@types/react-datepicker": "6.2.0",
68
- "@types/react-dom": "18.3.3",
68
+ "@types/react-dom": "18.3.5",
69
69
  "react": "18.3.1",
70
70
  "react-dom": "18.3.1",
71
71
  "@utils/test": "0.0.1"
@@ -85,7 +85,7 @@
85
85
  "react-toastify": "10.0.6",
86
86
  "react-use-clipboard": "1.0.9",
87
87
  "reakit": "1.3.11",
88
- "@ultraviolet/icons": "3.6.2",
88
+ "@ultraviolet/icons": "3.6.3",
89
89
  "@ultraviolet/themes": "1.15.0"
90
90
  },
91
91
  "scripts": {