UNPKG

@base-ui/react

Version:

Base UI is a library of headless ('unstyled') React components and low-level hooks. You gain complete control over your app's CSS and accessibility features.

260 lines (254 loc) 10.8 kB
'use client'; import * as React from 'react'; import { inertValue } from '@base-ui/utils/inertValue'; import { useIsoLayoutEffect } from '@base-ui/utils/useIsoLayoutEffect'; import { useTimeout } from '@base-ui/utils/useTimeout'; import { FloatingNode } from "../../floating-ui-react/index.js"; import { MenuPositionerContext } from "./MenuPositionerContext.js"; import { useMenuRootContext } from "../root/MenuRootContext.js"; import { useAnchorPositioning } from "../../utils/useAnchorPositioning.js"; import { CompositeList } from "../../internals/composite/list/CompositeList.js"; import { InternalBackdrop } from "../../utils/InternalBackdrop.js"; import { useMenuPortalContext } from "../portal/MenuPortalContext.js"; import { DROPDOWN_COLLISION_AVOIDANCE, POPUP_COLLISION_AVOIDANCE } from "../../internals/constants.js"; import { useContextMenuRootContext } from "../../context-menu/root/ContextMenuRootContext.js"; import { createChangeEventDetails } from "../../internals/createBaseUIEventDetails.js"; import { REASONS } from "../../internals/reasons.js"; import { adaptiveOrigin } from "../../utils/adaptiveOriginMiddleware.js"; import { useAnimationsFinished } from "../../internals/useAnimationsFinished.js"; import { usePositioner } from "../../utils/usePositioner.js"; import { useAnchoredPopupScrollLock } from "../../utils/useAnchoredPopupScrollLock.js"; /** * Positions the menu popup against the trigger. * Renders a `<div>` element. * * Documentation: [Base UI Menu](https://base-ui.com/react/components/menu) */ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime"; export const MenuPositioner = /*#__PURE__*/React.forwardRef(function MenuPositioner(componentProps, forwardedRef) { const { anchor: anchorProp, positionMethod: positionMethodProp = 'absolute', className, render, side, align: alignProp, sideOffset: sideOffsetProp = 0, alignOffset: alignOffsetProp = 0, collisionBoundary = 'clipping-ancestors', collisionPadding = 5, arrowPadding = 5, sticky = false, disableAnchorTracking = false, collisionAvoidance: collisionAvoidanceProp = DROPDOWN_COLLISION_AVOIDANCE, style, ...elementProps } = componentProps; const { store } = useMenuRootContext(); const keepMounted = useMenuPortalContext(); const contextMenuContext = useContextMenuRootContext(true); const parent = store.useState('parent'); const floatingRootContext = store.useState('floatingRootContext'); const floatingTreeRoot = store.useState('floatingTreeRoot'); const mounted = store.useState('mounted'); const open = store.useState('open'); const modal = store.useState('modal'); const openMethod = store.useState('openMethod'); const triggerElement = store.useState('activeTriggerElement'); const transitionStatus = store.useState('transitionStatus'); const positionerElement = store.useState('positionerElement'); const instantType = store.useState('instantType'); const hasViewport = store.useState('hasViewport'); const lastOpenChangeReason = store.useState('lastOpenChangeReason'); const floatingNodeId = store.useState('floatingNodeId'); const floatingParentNodeId = store.useState('floatingParentNodeId'); const domReference = floatingRootContext.useState('domReferenceElement'); const previousTriggerRef = React.useRef(null); const runOnceAnimationsFinish = useAnimationsFinished(positionerElement, false, false); let anchor = anchorProp; let sideOffset = sideOffsetProp; let alignOffset = alignOffsetProp; let align = alignProp; let collisionAvoidance = collisionAvoidanceProp; if (parent.type === 'context-menu') { anchor = anchorProp ?? parent.context?.anchor; align = align ?? 'start'; if (!side && align !== 'center') { alignOffset = componentProps.alignOffset ?? 2; sideOffset = componentProps.sideOffset ?? -5; } } let computedSide = side; let computedAlign = align; if (parent.type === 'menu') { computedSide = computedSide ?? 'inline-end'; computedAlign = computedAlign ?? 'start'; collisionAvoidance = componentProps.collisionAvoidance ?? POPUP_COLLISION_AVOIDANCE; } else if (parent.type === 'menubar') { computedSide = computedSide ?? 'bottom'; computedAlign = computedAlign ?? 'start'; } const contextMenu = parent.type === 'context-menu'; const positioner = useAnchorPositioning({ anchor, floatingRootContext, positionMethod: contextMenuContext ? 'fixed' : positionMethodProp, mounted, side: computedSide, sideOffset, align: computedAlign, alignOffset, arrowPadding: contextMenu ? 0 : arrowPadding, collisionBoundary, collisionPadding, sticky, nodeId: floatingNodeId, keepMounted, disableAnchorTracking, collisionAvoidance, shiftCrossAxis: contextMenu && !('side' in collisionAvoidance && collisionAvoidance.side === 'flip'), externalTree: floatingTreeRoot, adaptiveOrigin: hasViewport ? adaptiveOrigin : undefined }); React.useEffect(() => { function onMenuOpenChange(details) { if (details.open) { if (details.parentNodeId === floatingNodeId) { store.set('hoverEnabled', false); } if (details.nodeId !== floatingNodeId && details.parentNodeId === store.select('floatingParentNodeId')) { store.setOpen(false, createChangeEventDetails(REASONS.siblingOpen)); } } } floatingTreeRoot.events.on('menuopenchange', onMenuOpenChange); return () => { floatingTreeRoot.events.off('menuopenchange', onMenuOpenChange); }; }, [store, floatingTreeRoot.events, floatingNodeId]); React.useEffect(() => { if (store.select('floatingParentNodeId') == null) { return undefined; } function onParentClose(details) { if (details.open || details.nodeId !== store.select('floatingParentNodeId')) { return; } const reason = details.reason ?? REASONS.siblingOpen; store.setOpen(false, createChangeEventDetails(reason)); } floatingTreeRoot.events.on('menuopenchange', onParentClose); return () => { floatingTreeRoot.events.off('menuopenchange', onParentClose); }; }, [floatingTreeRoot.events, store]); const closeTimeout = useTimeout(); // Clear pending close timeout when the menu closes. React.useEffect(() => { if (!open) { closeTimeout.clear(); } }, [open, closeTimeout]); // Close unrelated child submenus when hovering a different item in the parent menu. React.useEffect(() => { function onItemHover(event) { // If an item within our parent menu is hovered, and this menu's trigger is not that item, // close this submenu. This ensures hovering a different item in the parent closes other branches. if (!open || event.nodeId !== store.select('floatingParentNodeId')) { return; } if (event.target && triggerElement && triggerElement !== event.target) { const delay = store.select('closeDelay'); if (delay > 0) { if (!closeTimeout.isStarted()) { closeTimeout.start(delay, () => { store.setOpen(false, createChangeEventDetails(REASONS.siblingOpen)); }); } } else { store.setOpen(false, createChangeEventDetails(REASONS.siblingOpen)); } } else { // User re-hovered the submenu trigger, cancel pending close. closeTimeout.clear(); } } floatingTreeRoot.events.on('itemhover', onItemHover); return () => { floatingTreeRoot.events.off('itemhover', onItemHover); }; }, [floatingTreeRoot.events, open, triggerElement, store, closeTimeout]); React.useEffect(() => { const eventDetails = { open, nodeId: floatingNodeId, parentNodeId: floatingParentNodeId, reason: store.select('lastOpenChangeReason') }; floatingTreeRoot.events.emit('menuopenchange', eventDetails); }, [floatingTreeRoot.events, open, store, floatingNodeId, floatingParentNodeId]); // Keep positioner transition behavior aligned with Popover when switching detached triggers. useIsoLayoutEffect(() => { const currentTrigger = domReference; const previousTrigger = previousTriggerRef.current; if (currentTrigger) { previousTriggerRef.current = currentTrigger; } if (previousTrigger && currentTrigger && currentTrigger !== previousTrigger) { store.set('instantType', undefined); const abortController = new AbortController(); runOnceAnimationsFinish(() => { store.set('instantType', 'trigger-change'); }, abortController.signal); return () => { abortController.abort(); }; } return undefined; }, [domReference, runOnceAnimationsFinish, store]); const state = { open, side: positioner.side, align: positioner.align, anchorHidden: positioner.anchorHidden, nested: parent.type === 'menu', instant: instantType }; const menubarModal = parent.type === 'menubar' && parent.context.modal; const popupModal = modal && lastOpenChangeReason !== REASONS.triggerHover; useAnchoredPopupScrollLock(open && (menubarModal || popupModal), openMethod === 'touch', positionerElement, triggerElement); const element = usePositioner(componentProps, state, { styles: positioner.positionerStyles, transitionStatus, props: elementProps, refs: [forwardedRef, store.useStateSetter('positionerElement')], hidden: !mounted, inert: !open }); const shouldRenderBackdrop = mounted && parent.type !== 'menu' && (parent.type !== 'menubar' && modal && lastOpenChangeReason !== REASONS.triggerHover || parent.type === 'menubar' && parent.context.modal); // cuts a hole in the backdrop to allow pointer interaction with the menubar or dropdown menu trigger element let backdropCutout = null; if (parent.type === 'menubar') { backdropCutout = parent.context.contentElement; } else if (parent.type === undefined) { backdropCutout = triggerElement; } return /*#__PURE__*/_jsxs(MenuPositionerContext.Provider, { value: positioner, children: [shouldRenderBackdrop && /*#__PURE__*/_jsx(InternalBackdrop, { ref: parent.type === 'context-menu' || parent.type === 'nested-context-menu' ? parent.context.internalBackdropRef : null, inert: inertValue(!open), cutout: backdropCutout }), /*#__PURE__*/_jsx(FloatingNode, { id: floatingNodeId, children: /*#__PURE__*/_jsx(CompositeList, { elementsRef: store.context.itemDomElements, labelsRef: store.context.itemLabels, children: element }) })] }); }); if (process.env.NODE_ENV !== "production") MenuPositioner.displayName = "MenuPositioner";