@codecademy/gamut
68.3.068.3.1-alpha.910133.0
dist/PopoverContainer/PopoverContainer.js~
dist/PopoverContainer/PopoverContainer.jsModified+9−9
Index: package/dist/PopoverContainer/PopoverContainer.js
===================================================================
--- package/dist/PopoverContainer/PopoverContainer.js
+++ package/dist/PopoverContainer/PopoverContainer.js
@@ -5,9 +5,9 @@
import * as React from 'react';
import { useWindowScroll, useWindowSize } from 'react-use';
import { BodyPortal } from '../BodyPortal';
import { FocusTrap } from '../FocusTrap';
-import { useResizingParentEffect, useScrollingParents, useScrollingParentsEffect } from './hooks';
+import { getRefElement, getTargetAsElement, useResizingParentEffect, useScrollingParents, useScrollingParentsEffect } from './hooks';
import { getContainers, getPosition, isOutOfView } from './utils';
import { jsx as _jsx } from "react/jsx-runtime";
const PopoverContent = /*#__PURE__*/_styled("div", {
target: "e1j9hq8r0",
@@ -15,9 +15,9 @@
})(variance.compose(system.positioning, variance.create({
transform: {
property: 'transform'
}
-})), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/PopoverContainer/PopoverContainer.tsx"],"names":[],"mappings":"AAiBuB","file":"../../src/PopoverContainer/PopoverContainer.tsx","sourcesContent":["import { elementDir, system, useElementDir } from '@codecademy/gamut-styles';\nimport { variance } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport * as React from 'react';\nimport { useWindowScroll, useWindowSize } from 'react-use';\n\nimport { BodyPortal } from '../BodyPortal';\nimport { FocusTrap } from '../FocusTrap';\nimport {\n  useResizingParentEffect,\n  useScrollingParents,\n  useScrollingParentsEffect,\n} from './hooks';\nimport { ContainerState, PopoverContainerProps } from './types';\nimport { getContainers, getPosition, isOutOfView } from './utils';\n\nconst PopoverContent = styled.div(\n  variance.compose(\n    system.positioning,\n    variance.create({\n      transform: {\n        property: 'transform',\n      },\n    })\n  )\n);\n\nexport const PopoverContainer: React.FC<PopoverContainerProps> = ({\n  alignment = 'bottom-left',\n  offset = 20,\n  y = 0,\n  x = 0,\n  invertAxis,\n  inline = false,\n  isOpen,\n  onRequestClose,\n  targetRef,\n  allowPageInteraction,\n  closeOnViewportExit = false,\n  ...rest\n}) => {\n  const popoverRef = useRef<HTMLDivElement>(null);\n  const hasRequestedCloseRef = useRef(false);\n  const onRequestCloseRef = useRef(onRequestClose);\n  const { width: winW, height: winH } = useWindowSize();\n  const { x: winX, y: winY } = useWindowScroll();\n  const [containers, setContainers] = useState<ContainerState>();\n  const [targetRect, setTargetRect] = useState<DOMRect>();\n  const parent = containers?.parent;\n\n  // Memoize scrolling parents to avoid expensive DOM traversals\n  const scrollingParents = useScrollingParents(targetRef);\n\n  // Keep onRequestClose ref up to date\n  useEffect(() => {\n    onRequestCloseRef.current = onRequestClose;\n  }, [onRequestClose]);\n\n  const targetDir = useElementDir(targetRef);\n\n  const targetEl = targetRef?.current;\n  /**\n   * See {@link getContainers} for more information\n   * Inline: `layoutSource` is derived from `offsetParent`.\n   * Portal: i.e. isn't inline so we can use the target itself.\n   */\n  const layoutSource =\n    inline && targetEl?.offsetParent instanceof Element\n      ? targetEl.offsetParent\n      : targetEl;\n  const isRtl = inline\n    ? layoutSource instanceof Element && elementDir(layoutSource) === 'rtl'\n    : targetDir === 'rtl';\n\n  const popoverPosition = useMemo(() => {\n    if (parent !== undefined) {\n      return getPosition({\n        alignment,\n        container: parent,\n        invertAxis,\n        isRtl,\n        offset,\n        x,\n        y,\n      });\n    }\n    return { styles: {}, dirNeutralStyles: undefined };\n  }, [parent, x, y, offset, alignment, invertAxis, isRtl]);\n\n  useEffect(() => {\n    const target = targetRef?.current;\n    if (!target) return;\n    setContainers(getContainers(target, inline, { x: winX, y: winY }));\n  }, [targetRef, inline, winW, winH, winX, winY, targetRect]);\n\n  // Update target rectangle when window size/scroll changes\n  useEffect(() => {\n    setTargetRect(targetRef?.current?.getBoundingClientRect());\n  }, [targetRef, isOpen, winW, winH, winX, winY]);\n\n  // Update target rectangle when parent size/scroll changes\n  const updateTargetPosition = useCallback(\n    (rect?: DOMRect) => {\n      const target = targetRef?.current;\n      if (!target) return;\n\n      const newRect = rect || target.getBoundingClientRect();\n      setTargetRect(newRect);\n\n      const currentScrollX =\n        window.pageXOffset || document.documentElement.scrollLeft;\n      const currentScrollY =\n        window.pageYOffset || document.documentElement.scrollTop;\n\n      setContainers(\n        getContainers(target, inline, { x: currentScrollX, y: currentScrollY })\n      );\n    },\n    [targetRef, inline]\n  );\n\n  useScrollingParentsEffect(targetRef, updateTargetPosition);\n\n  useResizingParentEffect(targetRef, setTargetRect);\n\n  // Handle closeOnViewportExit with cached scrolling parents for performance\n  useEffect(() => {\n    if (!closeOnViewportExit) return;\n\n    const rect = targetRect || containers?.viewport;\n    if (!rect) return;\n\n    const isOut = isOutOfView(\n      rect,\n      targetRef?.current as HTMLElement,\n      scrollingParents\n    );\n\n    if (isOut && !hasRequestedCloseRef.current) {\n      hasRequestedCloseRef.current = true;\n      onRequestCloseRef.current?.();\n    } else if (!isOut) {\n      hasRequestedCloseRef.current = false;\n    }\n  }, [\n    targetRect,\n    containers?.viewport,\n    targetRef,\n    closeOnViewportExit,\n    scrollingParents,\n  ]);\n  /**\n   * Allows targetRef to be or contain a button that toggles the popover open and closed.\n   * Without this check it would toggle closed then back open immediately.\n   */\n  const handleClickOutside = useCallback(\n    (e: MouseEvent | TouchEvent) => {\n      const target = e.target as Node;\n      const targetElement = targetRef.current;\n\n      if (!targetElement) return;\n      if (targetElement.contains(target)) return;\n      if (popoverRef.current?.contains(target)) return;\n\n      // If we get here, it's a genuine outside click\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef]\n  );\n\n  /**\n   * Backup click outside handler for cases where FocusTrap detection might be interfered with\n   * by our own floating elements\n   */\n  const handleGlobalClickOutside = useCallback(\n    (e: MouseEvent) => {\n      const target = e.target as Node;\n      const targetElement = targetRef.current;\n\n      if (!targetElement || !isOpen) return;\n\n      if (\n        targetElement.contains(target) ||\n        popoverRef.current?.contains(target)\n      )\n        return;\n\n      // Check if the clicked element is within an Overlay component\n      const clickedElement = target as Element;\n      if (clickedElement.closest('[data-floating=\"overlay\"]')) {\n        return;\n      }\n\n      // Check if the clicked element is within another Popover or PopoverContainer\n      const isFloatingElement = clickedElement.closest(\n        '[data-floating=\"popover\"]'\n      );\n      if (\n        isFloatingElement &&\n        !popoverRef.current?.contains(isFloatingElement)\n      ) {\n        onRequestClose?.();\n        return;\n      }\n\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef, isOpen]\n  );\n\n  // Backup global click listener for when a Popover or PopoverContainer is open\n  useEffect(() => {\n    if (isOpen) {\n      // Use a small delay to ensure this doesn't interfere with the FocusTrap's own detection\n      const timeoutId = setTimeout(() => {\n        document.addEventListener('mousedown', handleGlobalClickOutside, true);\n      }, 50);\n\n      return () => {\n        clearTimeout(timeoutId);\n        document.removeEventListener(\n          'mousedown',\n          handleGlobalClickOutside,\n          true\n        );\n      };\n    }\n  }, [isOpen, handleGlobalClickOutside]);\n\n  if (!isOpen || !targetRef) return null;\n\n  const {\n    children,\n    style: restStyle,\n    ...restProps\n  } = rest as React.HTMLAttributes<HTMLDivElement>;\n\n  const { dirNeutralStyles, styles: placementStyles } = popoverPosition;\n\n  /**\n   * Non-empty `placementStyles` merged into inline `style`\n   * dirNeutralStyles` layered last, see {@link getPosition}\n   */\n  const placementStylesToMerge =\n    Object.keys(placementStyles).length > 0 ? placementStyles : null;\n\n  const hasMergedStyle = Boolean(\n    restStyle || dirNeutralStyles || placementStylesToMerge\n  );\n  const mergedStyle = hasMergedStyle\n    ? {\n        ...(typeof restStyle === 'object' && restStyle ? restStyle : {}),\n        ...(placementStylesToMerge ?? {}),\n        ...dirNeutralStyles,\n      }\n    : undefined;\n\n  const content = (\n    <FocusTrap\n      allowPageInteraction={inline || allowPageInteraction}\n      onClickOutside={handleClickOutside}\n      onEscapeKey={onRequestClose}\n    >\n      <PopoverContent\n        data-floating=\"popover\"\n        data-testid=\"popover-content-container\"\n        position=\"absolute\"\n        ref={popoverRef}\n        /* eslint-disable-next-line gamut/no-inline-style */\n        style={mergedStyle}\n        tabIndex={-1}\n        zIndex={inline ? 5 : 'initial'}\n        {...restProps}\n      >\n        {children}\n      </PopoverContent>\n    </FocusTrap>\n  );\n\n  if (inline) return content;\n\n  return <BodyPortal>{content}</BodyPortal>;\n};\n"]} */");
+})), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/PopoverContainer/PopoverContainer.tsx"],"names":[],"mappings":"AAmBuB","file":"../../src/PopoverContainer/PopoverContainer.tsx","sourcesContent":["import { elementDir, system, useElementDir } from '@codecademy/gamut-styles';\nimport { variance } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport * as React from 'react';\nimport { useWindowScroll, useWindowSize } from 'react-use';\n\nimport { BodyPortal } from '../BodyPortal';\nimport { FocusTrap } from '../FocusTrap';\nimport {\n  getRefElement,\n  getTargetAsElement,\n  useResizingParentEffect,\n  useScrollingParents,\n  useScrollingParentsEffect,\n} from './hooks';\nimport { ContainerState, PopoverContainerProps, TargetRef } from './types';\nimport { getContainers, getPosition, isOutOfView } from './utils';\n\nconst PopoverContent = styled.div(\n  variance.compose(\n    system.positioning,\n    variance.create({\n      transform: {\n        property: 'transform',\n      },\n    })\n  )\n);\n\nexport const PopoverContainer: React.FC<PopoverContainerProps> = ({\n  alignment = 'bottom-left',\n  offset = 20,\n  y = 0,\n  x = 0,\n  invertAxis,\n  inline = false,\n  isOpen,\n  onRequestClose,\n  targetRef,\n  allowPageInteraction,\n  closeOnViewportExit = false,\n  ...rest\n}) => {\n  const popoverRef = useRef<HTMLDivElement>(null);\n  const hasRequestedCloseRef = useRef(false);\n  const onRequestCloseRef = useRef(onRequestClose);\n  const { width: winW, height: winH } = useWindowSize();\n  const { x: winX, y: winY } = useWindowScroll();\n  const [containers, setContainers] = useState<ContainerState>();\n  const [targetRect, setTargetRect] = useState<DOMRect>();\n  const parent = containers?.parent;\n\n  // Memoize scrolling parents to avoid expensive DOM traversals\n  const scrollingParents = useScrollingParents(targetRef);\n\n  // Keep onRequestClose ref up to date\n  useEffect(() => {\n    onRequestCloseRef.current = onRequestClose;\n  }, [onRequestClose]);\n\n  const targetDir = useElementDir(targetRef);\n\n  const targetEl = targetRef?.current;\n  /**\n   * See {@link getContainers} for more information\n   * Inline: `layoutSource` is derived from `offsetParent`.\n   * Portal: i.e. isn't inline so we can use the target itself.\n   */\n  const layoutSource =\n    inline && targetEl?.offsetParent instanceof Element\n      ? targetEl.offsetParent\n      : targetEl;\n  const isRtl = inline\n    ? layoutSource instanceof Element && elementDir(layoutSource) === 'rtl'\n    : targetDir === 'rtl';\n\n  const popoverPosition = useMemo(() => {\n    if (parent !== undefined) {\n      return getPosition({\n        alignment,\n        container: parent,\n        invertAxis,\n        isRtl,\n        offset,\n        x,\n        y,\n      });\n    }\n    return { styles: {}, dirNeutralStyles: undefined };\n  }, [parent, x, y, offset, alignment, invertAxis, isRtl]);\n\n  useEffect(() => {\n    const target = getRefElement(targetRef);\n    if (!target) return;\n    setContainers(\n      getContainers(target as TargetRef, inline, { x: winX, y: winY })\n    );\n  }, [targetRef, inline, winW, winH, winX, winY, targetRect]);\n\n  // Update target rectangle when window size/scroll changes\n  useEffect(() => {\n    setTargetRect(getRefElement(targetRef)?.getBoundingClientRect());\n  }, [targetRef, isOpen, winW, winH, winX, winY]);\n\n  // Update target rectangle when parent size/scroll changes\n  const updateTargetPosition = useCallback(\n    (rect?: DOMRect) => {\n      const target = getRefElement(targetRef);\n      if (!target) return;\n\n      const newRect = rect || target.getBoundingClientRect();\n      setTargetRect(newRect);\n\n      const currentScrollX =\n        window.pageXOffset || document.documentElement.scrollLeft;\n      const currentScrollY =\n        window.pageYOffset || document.documentElement.scrollTop;\n\n      setContainers(\n        getContainers(target as TargetRef, inline, {\n          x: currentScrollX,\n          y: currentScrollY,\n        })\n      );\n    },\n    [targetRef, inline]\n  );\n\n  useScrollingParentsEffect(targetRef, updateTargetPosition);\n  useResizingParentEffect(targetRef, setTargetRect);\n\n  // Handle closeOnViewportExit with cached scrolling parents for performance\n  useEffect(() => {\n    if (!closeOnViewportExit) return;\n\n    const rect = targetRect || containers?.viewport;\n    if (!rect) return;\n\n    const isOut = isOutOfView(\n      rect,\n      getTargetAsElement(getRefElement(targetRef)) ?? undefined,\n      scrollingParents\n    );\n\n    if (isOut && !hasRequestedCloseRef.current) {\n      hasRequestedCloseRef.current = true;\n      onRequestCloseRef.current?.();\n    } else if (!isOut) {\n      hasRequestedCloseRef.current = false;\n    }\n  }, [\n    targetRect,\n    containers?.viewport,\n    targetRef,\n    closeOnViewportExit,\n    scrollingParents,\n  ]);\n  /**\n   * Allows targetRef to be or contain a button that toggles the popover open and closed.\n   * Without this check it would toggle closed then back open immediately.\n   */\n  const handleClickOutside = useCallback(\n    (e: MouseEvent | TouchEvent) => {\n      const target = e.target as Node;\n      const targetElement = getRefElement(targetRef);\n\n      if (!targetElement) return;\n      if (targetElement.contains(target)) return;\n      if (popoverRef.current?.contains(target)) return;\n\n      // If we get here, it's a genuine outside click\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef]\n  );\n\n  /**\n   * Backup click outside handler for cases where FocusTrap detection might be interfered with\n   * by our own floating elements\n   */\n  const handleGlobalClickOutside = useCallback(\n    (e: MouseEvent) => {\n      const target = e.target as Node;\n      const targetElement = getRefElement(targetRef);\n\n      if (!targetElement || !isOpen) return;\n\n      if (\n        targetElement.contains(target) ||\n        popoverRef.current?.contains(target)\n      )\n        return;\n\n      // Check if the clicked element is within an Overlay component\n      const clickedElement = target as Element;\n      if (clickedElement.closest('[data-floating=\"overlay\"]')) {\n        return;\n      }\n\n      // Check if the clicked element is within another Popover or PopoverContainer\n      const isFloatingElement = clickedElement.closest(\n        '[data-floating=\"popover\"]'\n      );\n      if (\n        isFloatingElement &&\n        !popoverRef.current?.contains(isFloatingElement)\n      ) {\n        onRequestClose?.();\n        return;\n      }\n\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef, isOpen]\n  );\n\n  // Backup global click listener for when a Popover or PopoverContainer is open\n  useEffect(() => {\n    if (isOpen) {\n      // Use a small delay to ensure this doesn't interfere with the FocusTrap's own detection\n      const timeoutId = setTimeout(() => {\n        document.addEventListener('mousedown', handleGlobalClickOutside, true);\n      }, 50);\n\n      return () => {\n        clearTimeout(timeoutId);\n        document.removeEventListener(\n          'mousedown',\n          handleGlobalClickOutside,\n          true\n        );\n      };\n    }\n  }, [isOpen, handleGlobalClickOutside]);\n\n  if (!isOpen || !targetRef.current) return null;\n\n  const {\n    children,\n    style: restStyle,\n    ...restProps\n  } = rest as React.HTMLAttributes<HTMLDivElement>;\n\n  const { dirNeutralStyles, styles: placementStyles } = popoverPosition;\n\n  /**\n   * Non-empty `placementStyles` merged into inline `style`\n   * dirNeutralStyles` layered last, see {@link getPosition}\n   */\n  const placementStylesToMerge =\n    Object.keys(placementStyles).length > 0 ? placementStyles : null;\n\n  const hasMergedStyle = Boolean(\n    restStyle || dirNeutralStyles || placementStylesToMerge\n  );\n  const mergedStyle = hasMergedStyle\n    ? {\n        ...(typeof restStyle === 'object' && restStyle ? restStyle : {}),\n        ...(placementStylesToMerge ?? {}),\n        ...dirNeutralStyles,\n      }\n    : undefined;\n\n  const content = (\n    <FocusTrap\n      allowPageInteraction={inline || allowPageInteraction}\n      onClickOutside={handleClickOutside}\n      onEscapeKey={onRequestClose}\n    >\n      <PopoverContent\n        data-floating=\"popover\"\n        data-testid=\"popover-content-container\"\n        position=\"absolute\"\n        ref={popoverRef}\n        /* eslint-disable-next-line gamut/no-inline-style */\n        style={mergedStyle}\n        tabIndex={-1}\n        zIndex={inline ? 5 : 'initial'}\n        {...restProps}\n      >\n        {children}\n      </PopoverContent>\n    </FocusTrap>\n  );\n\n  if (inline) return content;\n\n  return <BodyPortal>{content}</BodyPortal>;\n};\n"]} */");
export const PopoverContainer = ({
alignment = 'bottom-left',
offset = 20,
y = 0,
@@ -79,9 +79,9 @@
dirNeutralStyles: undefined
};
}, [parent, x, y, offset, alignment, invertAxis, isRtl]);
useEffect(() => {
- const target = targetRef?.current;
+ const target = getRefElement(targetRef);
if (!target) return;
setContainers(getContainers(target, inline, {
x: winX,
y: winY
@@ -89,14 +89,14 @@
}, [targetRef, inline, winW, winH, winX, winY, targetRect]);
// Update target rectangle when window size/scroll changes
useEffect(() => {
- setTargetRect(targetRef?.current?.getBoundingClientRect());
+ setTargetRect(getRefElement(targetRef)?.getBoundingClientRect());
}, [targetRef, isOpen, winW, winH, winX, winY]);
// Update target rectangle when parent size/scroll changes
const updateTargetPosition = useCallback(rect => {
- const target = targetRef?.current;
+ const target = getRefElement(targetRef);
if (!target) return;
const newRect = rect || target.getBoundingClientRect();
setTargetRect(newRect);
const currentScrollX = window.pageXOffset || document.documentElement.scrollLeft;
@@ -113,9 +113,9 @@
useEffect(() => {
if (!closeOnViewportExit) return;
const rect = targetRect || containers?.viewport;
if (!rect) return;
- const isOut = isOutOfView(rect, targetRef?.current, scrollingParents);
+ const isOut = isOutOfView(rect, getTargetAsElement(getRefElement(targetRef)) ?? undefined, scrollingParents);
if (isOut && !hasRequestedCloseRef.current) {
hasRequestedCloseRef.current = true;
onRequestCloseRef.current?.();
} else if (!isOut) {
@@ -127,9 +127,9 @@
* Without this check it would toggle closed then back open immediately.
*/
const handleClickOutside = useCallback(e => {
const target = e.target;
- const targetElement = targetRef.current;
+ const targetElement = getRefElement(targetRef);
if (!targetElement) return;
if (targetElement.contains(target)) return;
if (popoverRef.current?.contains(target)) return;
@@ -142,9 +142,9 @@
* by our own floating elements
*/
const handleGlobalClickOutside = useCallback(e => {
const target = e.target;
- const targetElement = targetRef.current;
+ const targetElement = getRefElement(targetRef);
if (!targetElement || !isOpen) return;
if (targetElement.contains(target) || popoverRef.current?.contains(target)) return;
// Check if the clicked element is within an Overlay component
@@ -174,9 +174,9 @@
document.removeEventListener('mousedown', handleGlobalClickOutside, true);
};
}
}, [isOpen, handleGlobalClickOutside]);
- if (!isOpen || !targetRef) return null;
+ if (!isOpen || !targetRef.current) return null;
const {
children,
style: restStyle,
...restProps