@amboss/design-system 1.20.4 → 1.21.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (44) hide show
  1. package/build/cjs/build-tokens/visualConfig.js +11 -0
  2. package/build/cjs/src/components/Popover/Popover.js +8 -3
  3. package/build/cjs/src/components/Sheet/Sheet.js +315 -0
  4. package/build/cjs/src/components/Toggletip/BasePopover.js +43 -8
  5. package/build/cjs/src/components/Tooltip/BaseTooltip.js +1 -1
  6. package/build/cjs/src/components/Tooltip/TooltipContent.js +41 -22
  7. package/build/cjs/src/components/UserHighlightTooltip/UserHighlightTooltip.js +1 -1
  8. package/build/cjs/src/index.js +2 -0
  9. package/build/cjs/src/shared/FocusTrapWrapper.js +41 -0
  10. package/build/cjs/src/shared/useDragDown.js +84 -0
  11. package/build/esm/build-tokens/_colors.json +9 -0
  12. package/build/esm/build-tokens/_sizes.json +1 -1
  13. package/build/esm/build-tokens/visualConfig.d.ts +8 -0
  14. package/build/esm/build-tokens/visualConfig.js +11 -0
  15. package/build/esm/build-tokens/visualConfig.js.map +1 -1
  16. package/build/esm/src/components/Popover/Popover.d.ts +2 -3
  17. package/build/esm/src/components/Popover/Popover.js +8 -3
  18. package/build/esm/src/components/Popover/Popover.js.map +1 -1
  19. package/build/esm/src/components/Sheet/Sheet.d.ts +15 -0
  20. package/build/esm/src/components/Sheet/Sheet.js +308 -0
  21. package/build/esm/src/components/Sheet/Sheet.js.map +1 -0
  22. package/build/esm/src/components/Toggletip/BasePopover.d.ts +5 -5
  23. package/build/esm/src/components/Toggletip/BasePopover.js +43 -7
  24. package/build/esm/src/components/Toggletip/BasePopover.js.map +1 -1
  25. package/build/esm/src/components/Toggletip/Toggletip.d.ts +1 -1
  26. package/build/esm/src/components/Toggletip/Toggletip.js.map +1 -1
  27. package/build/esm/src/components/Tooltip/BaseTooltip.js +1 -1
  28. package/build/esm/src/components/Tooltip/BaseTooltip.js.map +1 -1
  29. package/build/esm/src/components/Tooltip/TooltipContent.js +42 -23
  30. package/build/esm/src/components/Tooltip/TooltipContent.js.map +1 -1
  31. package/build/esm/src/components/UserHighlightTooltip/UserHighlightTooltip.js +1 -1
  32. package/build/esm/src/components/UserHighlightTooltip/UserHighlightTooltip.js.map +1 -1
  33. package/build/esm/src/index.d.ts +1 -0
  34. package/build/esm/src/index.js +1 -0
  35. package/build/esm/src/index.js.map +1 -1
  36. package/build/esm/src/shared/FocusTrapWrapper.d.ts +10 -0
  37. package/build/esm/src/shared/FocusTrapWrapper.js +35 -0
  38. package/build/esm/src/shared/FocusTrapWrapper.js.map +1 -0
  39. package/build/esm/src/shared/useDragDown.d.ts +13 -0
  40. package/build/esm/src/shared/useDragDown.js +83 -0
  41. package/build/esm/src/shared/useDragDown.js.map +1 -0
  42. package/build/scss/_theming.scss +2 -0
  43. package/build/scss/_variables.scss +1 -0
  44. package/package.json +1 -1
@@ -9,7 +9,6 @@ var useWindow = require('../../shared/useWindow.js');
9
9
  var SubThemeProvider = require('../SubThemeProvider/SubThemeProvider.js');
10
10
  var _zindex = require('../../../build-tokens/_zindex.json.js');
11
11
  var utils = require('./utils.js');
12
- var Container = require('../Container/Container.js');
13
12
 
14
13
  function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
15
14
 
@@ -48,9 +47,7 @@ const StyledContainer = /*#__PURE__*/_styled__default.default("div", process.env
48
47
  padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,
49
48
  ...(contentPadding && {
50
49
  padding: contentPaddingMap[contentPadding]
51
- }),
52
- borderRadius: theme.variables.size.borderRadius.xs,
53
- backgroundColor: theme.values.color.background.primary.default
50
+ })
54
51
  };
55
52
  return {
56
53
  position: "absolute",
@@ -58,14 +55,21 @@ const StyledContainer = /*#__PURE__*/_styled__default.default("div", process.env
58
55
  opacity: 0,
59
56
  animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,
60
57
  maxWidth,
61
- width: "initial",
58
+ width: "max-content",
62
59
  boxSizing: "border-box",
60
+ backgroundColor: hasInvertedSubTheme ? theme.values.color.background.primary.default : theme.values.color.background.elevated.default,
61
+ borderRadius: hasInvertedSubTheme ? theme.variables.size.borderRadius.xs : theme.variables.size.borderRadius.s,
63
62
  ...(hasInvertedSubTheme && invertedSubThemeStyles),
64
63
  ...(horizontalPlacement === "center" && {
65
64
  transform: "translate(-50%)"
65
+ }),
66
+ ...(!hasInvertedSubTheme && {
67
+ boxShadow: theme.values.elevation.c
66
68
  })
67
69
  };
68
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TooltipContent.tsx"],"names":[],"mappings":"AA6EwB","file":"TooltipContent.tsx","sourcesContent":["import React, {\n  MutableRefObject,\n  ReactElement,\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\nimport { Container } from \"../Container/Container\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n      borderRadius: theme.variables.size.borderRadius.xs,\n      backgroundColor: theme.values.color.background.primary.default,\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"initial\",\n      boxSizing: \"border-box\",\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({ theme, verticalPlacement, horizontalPlacement, size = ARROW_SIZE }) => {\n    const offset = getArrowOffset(size);\n    return {\n      position: \"absolute\",\n      width: 0,\n      height: 0,\n      borderLeft: `${size}px solid transparent`,\n      borderRight: `${size}px solid transparent`,\n\n      ...(verticalPlacement === \"top\" && {\n        top: \"100%\",\n        borderTop: `${size}px solid ${theme.values.color.background.primary.default}`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        top: `-${size}px`,\n        borderBottom: `${size}px solid ${theme.values.color.background.primary.default}`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset}px`,\n      }),\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  // Re-position tooltip if it moves out of the viewport by 10% and on window resize\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      isVisible &&\n      tooltipRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach(() => {\n            calculateStyle();\n          });\n        },\n        {\n          threshold: 0.9,\n        }\n      );\n\n      observer.observe(tooltipRef.current);\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {hasInvertedSubTheme ? (\n        content\n      ) : (\n        <Container elevation={3} borderRadius=\"s\">\n          {content}\n        </Container>\n      )}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"]} */");
70
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TooltipContent.tsx"],"names":[],"mappings":"AA6EwB","file":"TooltipContent.tsx","sourcesContent":["import React, {\n  MutableRefObject,\n  ReactElement,\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  DISTANCE_FROM_TRIGGER,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: hasInvertedSubTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: hasInvertedSubTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n      ...(!hasInvertedSubTheme && {\n        boxShadow: theme.values.elevation.c,\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = Pick<TooltipContentProps, \"hasInvertedSubTheme\"> & {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\n// This container is large enough to contain the arrow shadow blur\nconst ARROW_CONTAINER_WIDTH = 40;\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({\n    theme,\n    hasInvertedSubTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = hasInvertedSubTheme ? 0 : 1;\n    const arrowContainerHeight = size + DISTANCE_FROM_TRIGGER;\n    // Get arrow width and height using pythogoras theorem and add 1 to height to account for dark mode shadow.\n    const arrowSideLength = Math.sqrt(\n      size ** 2 + (size + adjustmentForShadow) ** 2\n    );\n\n    return {\n      position: \"absolute\",\n      width: ARROW_CONTAINER_WIDTH,\n      height: arrowContainerHeight,\n      overflow: \"hidden\",\n\n      ...(verticalPlacement === \"top\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `calc(100% - ${adjustmentForShadow}px)`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `-${arrowContainerHeight - adjustmentForShadow}px`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      \"&::after\": {\n        content: '\" \"',\n        position: \"absolute\",\n        top: verticalPlacement === \"top\" ? 0 : \"100%\",\n        left: \"50%\",\n        width: arrowSideLength,\n        height: arrowSideLength,\n        backgroundColor: hasInvertedSubTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!hasInvertedSubTheme && {\n          boxShadow: theme.values.elevation.c,\n        }),\n      },\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  // Re-position tooltip if it moves out of the viewport by 10% and on window resize\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      isVisible &&\n      tooltipRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach(() => {\n            calculateStyle();\n          });\n        },\n        {\n          threshold: 0.9,\n        }\n      );\n\n      observer.observe(tooltipRef.current);\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          hasInvertedSubTheme={hasInvertedSubTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"]} */");
71
+ // This container is large enough to contain the arrow shadow blur
72
+ const ARROW_CONTAINER_WIDTH = 40;
69
73
  const StyledArrow = /*#__PURE__*/_styled__default.default("div", process.env.NODE_ENV === "production" ? {
70
74
  target: "e1k814k40"
71
75
  } : {
@@ -74,37 +78,54 @@ const StyledArrow = /*#__PURE__*/_styled__default.default("div", process.env.NOD
74
78
  })(_ref2 => {
75
79
  let {
76
80
  theme,
81
+ hasInvertedSubTheme,
77
82
  verticalPlacement,
78
83
  horizontalPlacement,
79
84
  size = utils.ARROW_SIZE
80
85
  } = _ref2;
81
86
  const offset = utils.getArrowOffset(size);
87
+ const adjustmentForShadow = hasInvertedSubTheme ? 0 : 1;
88
+ const arrowContainerHeight = size + utils.DISTANCE_FROM_TRIGGER;
89
+ // Get arrow width and height using pythogoras theorem and add 1 to height to account for dark mode shadow.
90
+ const arrowSideLength = Math.sqrt(size ** 2 + (size + adjustmentForShadow) ** 2);
82
91
  return {
83
92
  position: "absolute",
84
- width: 0,
85
- height: 0,
86
- borderLeft: `${size}px solid transparent`,
87
- borderRight: `${size}px solid transparent`,
93
+ width: ARROW_CONTAINER_WIDTH,
94
+ height: arrowContainerHeight,
95
+ overflow: "hidden",
88
96
  ...(verticalPlacement === "top" && {
89
- top: "100%",
90
- borderTop: `${size}px solid ${theme.values.color.background.primary.default}`
97
+ // place the arrow container 1px inside tooltip container to account for dark mode box-shadow
98
+ top: `calc(100% - ${adjustmentForShadow}px)`
91
99
  }),
92
100
  ...(verticalPlacement === "bottom" && {
93
- top: `-${size}px`,
94
- borderBottom: `${size}px solid ${theme.values.color.background.primary.default}`
101
+ // place the arrow container 1px inside tooltip container to account for dark mode box-shadow
102
+ top: `-${arrowContainerHeight - adjustmentForShadow}px`
95
103
  }),
96
104
  ...(horizontalPlacement === "center" && {
97
105
  left: "50%",
98
106
  transform: "translate(-50%)"
99
107
  }),
100
108
  ...(horizontalPlacement === "right" && {
101
- left: `${offset}px`
109
+ left: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`
102
110
  }),
103
111
  ...(horizontalPlacement === "left" && {
104
- right: `${offset}px`
105
- })
112
+ right: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`
113
+ }),
114
+ "&::after": {
115
+ content: '" "',
116
+ position: "absolute",
117
+ top: verticalPlacement === "top" ? 0 : "100%",
118
+ left: "50%",
119
+ width: arrowSideLength,
120
+ height: arrowSideLength,
121
+ backgroundColor: hasInvertedSubTheme ? theme.values.color.background.primary.default : theme.values.color.background.elevated.default,
122
+ transform: "translate(-50%, -50%) rotate(45deg)",
123
+ ...(!hasInvertedSubTheme && {
124
+ boxShadow: theme.values.elevation.c
125
+ })
126
+ }
106
127
  };
107
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TooltipContent.tsx"],"names":[],"mappings":"AAwIoB","file":"TooltipContent.tsx","sourcesContent":["import React, {\n  MutableRefObject,\n  ReactElement,\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\nimport { Container } from \"../Container/Container\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n      borderRadius: theme.variables.size.borderRadius.xs,\n      backgroundColor: theme.values.color.background.primary.default,\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"initial\",\n      boxSizing: \"border-box\",\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({ theme, verticalPlacement, horizontalPlacement, size = ARROW_SIZE }) => {\n    const offset = getArrowOffset(size);\n    return {\n      position: \"absolute\",\n      width: 0,\n      height: 0,\n      borderLeft: `${size}px solid transparent`,\n      borderRight: `${size}px solid transparent`,\n\n      ...(verticalPlacement === \"top\" && {\n        top: \"100%\",\n        borderTop: `${size}px solid ${theme.values.color.background.primary.default}`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        top: `-${size}px`,\n        borderBottom: `${size}px solid ${theme.values.color.background.primary.default}`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset}px`,\n      }),\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  // Re-position tooltip if it moves out of the viewport by 10% and on window resize\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      isVisible &&\n      tooltipRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach(() => {\n            calculateStyle();\n          });\n        },\n        {\n          threshold: 0.9,\n        }\n      );\n\n      observer.observe(tooltipRef.current);\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {hasInvertedSubTheme ? (\n        content\n      ) : (\n        <Container elevation={3} borderRadius=\"s\">\n          {content}\n        </Container>\n      )}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"]} */");
128
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TooltipContent.tsx"],"names":[],"mappings":"AAmJoB","file":"TooltipContent.tsx","sourcesContent":["import React, {\n  MutableRefObject,\n  ReactElement,\n  useCallback,\n  useEffect,\n  useLayoutEffect,\n  useMemo,\n  useRef,\n  useState,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ANIMATION_DISTANCE,\n  ARROW_SIZE,\n  ARROW_SIZE_BIG,\n  DISTANCE_FROM_TRIGGER,\n  getArrowOffset,\n  getTooltipStyle,\n} from \"./utils\";\n\nexport type TooltipContentProps = {\n  content: ReactElement;\n  triggerRef: MutableRefObject<any>;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  dataE2eTestId?: string;\n  dataDSId: string;\n  isVisible?: boolean;\n  tooltipId?: string;\n  \"aria-hidden\"?: boolean;\n  role?: string;\n  tabIndex?: number;\n  // Content padding\n  contentPadding?: \"s\" | \"m\";\n  hasInvertedSubTheme?: boolean;\n  maxWidth?: number;\n  defaultVerticalPlacement?: TooltipStyle[\"verticalPlacement\"];\n  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\n  hideArrow?: boolean;\n};\n\nexport type TooltipStyle = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = Pick<\n  TooltipContentProps,\n  \"contentPadding\" | \"maxWidth\" | \"hasInvertedSubTheme\"\n> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\nconst MAX_CONTENT_WIDTH = 224;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({\n    theme,\n    horizontalPlacement,\n    verticalPlacement,\n    maxWidth,\n    contentPadding,\n    hasInvertedSubTheme = true,\n  }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    const contentPaddingMap = {\n      s: theme.variables.size.spacing.xxs,\n      m: theme.variables.size.spacing.s,\n    };\n\n    const invertedSubThemeStyles = {\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n    };\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      maxWidth,\n      width: \"max-content\",\n      boxSizing: \"border-box\",\n      backgroundColor: hasInvertedSubTheme\n        ? theme.values.color.background.primary.default\n        : theme.values.color.background.elevated.default,\n      borderRadius: hasInvertedSubTheme\n        ? theme.variables.size.borderRadius.xs\n        : theme.variables.size.borderRadius.s,\n\n      ...(hasInvertedSubTheme && invertedSubThemeStyles),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n      ...(!hasInvertedSubTheme && {\n        boxShadow: theme.values.elevation.c,\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = Pick<TooltipContentProps, \"hasInvertedSubTheme\"> & {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  size?: 0 | typeof ARROW_SIZE | typeof ARROW_SIZE_BIG;\n};\n\n// This container is large enough to contain the arrow shadow blur\nconst ARROW_CONTAINER_WIDTH = 40;\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({\n    theme,\n    hasInvertedSubTheme,\n    verticalPlacement,\n    horizontalPlacement,\n    size = ARROW_SIZE,\n  }) => {\n    const offset = getArrowOffset(size);\n    const adjustmentForShadow = hasInvertedSubTheme ? 0 : 1;\n    const arrowContainerHeight = size + DISTANCE_FROM_TRIGGER;\n    // Get arrow width and height using pythogoras theorem and add 1 to height to account for dark mode shadow.\n    const arrowSideLength = Math.sqrt(\n      size ** 2 + (size + adjustmentForShadow) ** 2\n    );\n\n    return {\n      position: \"absolute\",\n      width: ARROW_CONTAINER_WIDTH,\n      height: arrowContainerHeight,\n      overflow: \"hidden\",\n\n      ...(verticalPlacement === \"top\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `calc(100% - ${adjustmentForShadow}px)`,\n      }),\n\n      ...(verticalPlacement === \"bottom\" && {\n        // place the arrow container 1px inside tooltip container to account for dark mode box-shadow\n        top: `-${arrowContainerHeight - adjustmentForShadow}px`,\n      }),\n\n      ...(horizontalPlacement === \"center\" && {\n        left: \"50%\",\n        transform: \"translate(-50%)\",\n      }),\n\n      ...(horizontalPlacement === \"right\" && {\n        left: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      ...(horizontalPlacement === \"left\" && {\n        right: `${offset - (ARROW_CONTAINER_WIDTH / 2 - size)}px`,\n      }),\n\n      \"&::after\": {\n        content: '\" \"',\n        position: \"absolute\",\n        top: verticalPlacement === \"top\" ? 0 : \"100%\",\n        left: \"50%\",\n        width: arrowSideLength,\n        height: arrowSideLength,\n        backgroundColor: hasInvertedSubTheme\n          ? theme.values.color.background.primary.default\n          : theme.values.color.background.elevated.default,\n        transform: \"translate(-50%, -50%) rotate(45deg)\",\n\n        ...(!hasInvertedSubTheme && {\n          boxShadow: theme.values.elevation.c,\n        }),\n      },\n    };\n  }\n);\n\nconst initialStyle: TooltipStyle = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\n/** This component is used to display the overlay for both Toggletip and Tooltip components */\nexport function TooltipContent({\n  placement = \"auto\",\n  content,\n  tooltipId,\n  triggerRef,\n  portalContainer,\n  dataE2eTestId,\n  dataDSId,\n  isVisible,\n  \"aria-hidden\": ariaHidden,\n  role,\n  tabIndex,\n  contentPadding,\n  maxWidth = MAX_CONTENT_WIDTH,\n  hasInvertedSubTheme = true,\n  defaultVerticalPlacement,\n  onTooltipPointerEnter,\n  onTooltipPointerLeave,\n  hideArrow = false,\n}: TooltipContentProps): React.ReactElement {\n  const [style, setStyle] = useState(initialStyle);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n\n  const arrowSize = useMemo(() => {\n    if (hideArrow) {\n      return 0;\n    }\n    return hasInvertedSubTheme ? ARROW_SIZE : ARROW_SIZE_BIG;\n  }, [hasInvertedSubTheme, hideArrow]);\n\n  const calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(\n          placement,\n          defaultVerticalPlacement,\n          triggerRef,\n          tooltipRef,\n          document,\n          window,\n          arrowSize\n        )\n      );\n    }\n  }, [\n    triggerRef,\n    tooltipRef,\n    document,\n    window,\n    placement,\n    arrowSize,\n    defaultVerticalPlacement,\n  ]);\n\n  // This layout effect to re-render with updated position after determining content width\n  useLayoutEffect(() => {\n    if (isVisible) {\n      calculateStyle();\n    }\n  }, [isVisible, calculateStyle, contentPadding, content]);\n\n  // Re-position tooltip if it moves out of the viewport by 10% and on window resize\n  useEffect(() => {\n    let observer: IntersectionObserver;\n\n    if (\n      typeof IntersectionObserver !== \"undefined\" &&\n      isVisible &&\n      tooltipRef.current\n    ) {\n      observer = new IntersectionObserver(\n        (entries) => {\n          entries.forEach(() => {\n            calculateStyle();\n          });\n        },\n        {\n          threshold: 0.9,\n        }\n      );\n\n      observer.observe(tooltipRef.current);\n      window.addEventListener(\"resize\", calculateStyle);\n      window.addEventListener(\"scroll\", calculateStyle, true); // use capture here to detect scroll on any parent\n    } else if (!isVisible) {\n      // log time when tooltip closes\n      lastTooltipHideTimestamp = Date.now();\n    }\n\n    return () => {\n      if (observer) {\n        observer.disconnect();\n      }\n      window.removeEventListener(\"resize\", calculateStyle);\n      window.removeEventListener(\"scroll\", calculateStyle);\n    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  if (!isVisible) return null;\n\n  const tooltipElm = (\n    <StyledContainer\n      data-e2e-test-id={dataE2eTestId}\n      data-ds-id={dataDSId}\n      style={{\n        top: style.top,\n        left: style.left,\n        animationDuration: getAnimationDuration(),\n      }}\n      ref={tooltipRef}\n      id={tooltipId}\n      role={role}\n      aria-hidden={ariaHidden}\n      hasInvertedSubTheme={hasInvertedSubTheme}\n      tabIndex={tabIndex}\n      horizontalPlacement={style.horizontalPlacement}\n      verticalPlacement={style.verticalPlacement}\n      maxWidth={maxWidth}\n      contentPadding={contentPadding}\n      onPointerEnter={onTooltipPointerEnter}\n      onPointerLeave={onTooltipPointerLeave}\n    >\n      {content}\n      {!hideArrow && (\n        <StyledArrow\n          data-e2e-test-id={`${dataE2eTestId}_arrow`}\n          hasInvertedSubTheme={hasInvertedSubTheme}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          size={arrowSize}\n        />\n      )}\n    </StyledContainer>\n  );\n\n  const wrapperElm = hasInvertedSubTheme ? (\n    <SubThemeProvider name=\"inverted\">{tooltipElm}</SubThemeProvider>\n  ) : (\n    tooltipElm\n  );\n\n  return createPortal(wrapperElm, portalContainer || document.body);\n}\n"]} */");
108
129
  const initialStyle = {
109
130
  top: 0,
110
131
  left: 0,
@@ -218,11 +239,9 @@ function TooltipContent(_ref3) {
218
239
  contentPadding: contentPadding,
219
240
  onPointerEnter: onTooltipPointerEnter,
220
241
  onPointerLeave: onTooltipPointerLeave
221
- }, hasInvertedSubTheme ? content : /*#__PURE__*/React__default.default.createElement(Container.Container, {
222
- elevation: 3,
223
- borderRadius: "s"
224
- }, content), !hideArrow && /*#__PURE__*/React__default.default.createElement(StyledArrow, {
242
+ }, content, !hideArrow && /*#__PURE__*/React__default.default.createElement(StyledArrow, {
225
243
  "data-e2e-test-id": `${dataE2eTestId}_arrow`,
244
+ hasInvertedSubTheme: hasInvertedSubTheme,
226
245
  horizontalPlacement: style.horizontalPlacement,
227
246
  verticalPlacement: style.verticalPlacement,
228
247
  size: arrowSize
@@ -11,7 +11,7 @@ const SHOW_HIDE_DELAY = 200;
11
11
  function UserHighlightTooltip(_ref) {
12
12
  let {
13
13
  content,
14
- contentPadding = "m",
14
+ contentPadding = "s",
15
15
  placement = "auto",
16
16
  maxWidth,
17
17
  portalContainer,
@@ -52,6 +52,7 @@ var TagGroup = require('./components/TagGroup/TagGroup.js');
52
52
  var Toggletip = require('./components/Toggletip/Toggletip.js');
53
53
  var Popover = require('./components/Popover/Popover.js');
54
54
  var UserHighlightTooltip = require('./components/UserHighlightTooltip/UserHighlightTooltip.js');
55
+ var Sheet = require('./components/Sheet/Sheet.js');
55
56
  var Input = require('./components/Form/Input/Input.js');
56
57
  var PasswordInput = require('./components/Form/PasswordInput/PasswordInput.js');
57
58
  var DataTable = require('./components/DataTable/DataTable.js');
@@ -137,6 +138,7 @@ exports.TagGroup = TagGroup.TagGroup;
137
138
  exports.Toggletip = Toggletip.Toggletip;
138
139
  exports.Popover = Popover.Popover;
139
140
  exports.UserHighlightTooltip = UserHighlightTooltip.UserHighlightTooltip;
141
+ exports.Sheet = Sheet.Sheet;
140
142
  exports.Input = Input.Input;
141
143
  exports.PasswordInput = PasswordInput.PasswordInput;
142
144
  exports.BaseDataTable = DataTable.BaseDataTable;
@@ -0,0 +1,41 @@
1
+ 'use strict';
2
+
3
+ var React = require('react');
4
+ var FocusTrap = require('focus-trap-react');
5
+
6
+ function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
7
+
8
+ var React__default = /*#__PURE__*/_interopDefault(React);
9
+ var FocusTrap__default = /*#__PURE__*/_interopDefault(FocusTrap);
10
+
11
+ /**
12
+ This wrapper is added to handle onPostDeactivate from react-focus-trap. onPostDeactivate is fired before component unmouns. Often, onDeactivate is used to unmount the component (see BasePopover and Sheet), so we don't want to handle this event again before component unmounts.
13
+ */
14
+
15
+ const FocusTrapWrapper = _ref => {
16
+ let {
17
+ focusTrapOptions,
18
+ children
19
+ } = _ref;
20
+ const [isDeactivated, setDeactivated] = React.useState(false);
21
+ const {
22
+ onPostDeactivate,
23
+ ...rest
24
+ } = focusTrapOptions;
25
+ const handleDeactivate = () => {
26
+ setDeactivated(true);
27
+ };
28
+ React.useEffect(() => {
29
+ if (isDeactivated) {
30
+ onPostDeactivate();
31
+ }
32
+ }, [isDeactivated, onPostDeactivate]);
33
+ return /*#__PURE__*/React__default.default.createElement(FocusTrap__default.default, {
34
+ focusTrapOptions: {
35
+ ...rest,
36
+ onDeactivate: handleDeactivate
37
+ }
38
+ }, children);
39
+ };
40
+
41
+ exports.FocusTrapWrapper = FocusTrapWrapper;
@@ -0,0 +1,84 @@
1
+ 'use strict';
2
+
3
+ var React = require('react');
4
+
5
+ /**
6
+ * Handle Drag down gesture for mobile web
7
+ */
8
+ function useDragDown(_ref) {
9
+ let {
10
+ ref,
11
+ isVisible = false,
12
+ onDragStart,
13
+ onDrag,
14
+ onDragEnd,
15
+ onDragCancel
16
+ } = _ref;
17
+ React.useEffect(() => {
18
+ let touchStartY = 0;
19
+ let touchMoveY = 0;
20
+ let touchEndY = 0;
21
+ let dragCancelled = false;
22
+ let elm;
23
+ const cancelDrag = () => {
24
+ dragCancelled = true;
25
+ touchStartY = 0;
26
+ touchMoveY = 0;
27
+ touchEndY = 0;
28
+ onDragCancel();
29
+ };
30
+ const handleTouchStart = evt => {
31
+ evt.stopPropagation();
32
+ touchStartY = evt.changedTouches[0].screenY;
33
+ touchMoveY = touchStartY;
34
+ onDragStart(touchStartY);
35
+ dragCancelled = false;
36
+ };
37
+ const handleTouchEnd = evt => {
38
+ evt.stopPropagation();
39
+ if (!dragCancelled) {
40
+ touchEndY = evt.changedTouches[0].screenY;
41
+ if (touchEndY >= touchMoveY) {
42
+ onDragEnd(touchEndY, touchEndY - touchStartY);
43
+ } else {
44
+ // drag ended in a different direction
45
+ cancelDrag();
46
+ }
47
+ }
48
+ };
49
+ const handleTouchMove = evt => {
50
+ evt.stopPropagation();
51
+ if (!dragCancelled) {
52
+ const prevTouchMoveY = touchMoveY;
53
+ touchMoveY = evt.changedTouches[0].screenY;
54
+ if (touchMoveY > prevTouchMoveY) {
55
+ onDrag(touchMoveY, touchMoveY - touchStartY);
56
+ } else {
57
+ // drag ended in a different direction
58
+ cancelDrag();
59
+ }
60
+ }
61
+ };
62
+ if (isVisible && ref.current) {
63
+ elm = ref.current;
64
+ elm.addEventListener("touchstart", handleTouchStart, {
65
+ passive: true
66
+ });
67
+ elm.addEventListener("touchend", handleTouchEnd, {
68
+ passive: true
69
+ });
70
+ elm.addEventListener("touchmove", handleTouchMove, {
71
+ passive: true
72
+ });
73
+ }
74
+ return () => {
75
+ if (elm) {
76
+ elm.removeEventListener("touchstart", handleTouchStart);
77
+ elm.removeEventListener("touchend", handleTouchEnd);
78
+ elm.removeEventListener("touchmove", handleTouchMove);
79
+ }
80
+ };
81
+ }, [ref, isVisible, onDragStart, onDragCancel, onDragEnd, onDrag]);
82
+ }
83
+
84
+ exports.useDragDown = useDragDown;
@@ -78,6 +78,15 @@
78
78
  "lightValue": "rgba(15, 169, 128, 0.3)",
79
79
  "lightOriginalValue": "green.regular-transparent"
80
80
  },
81
+ "color-background-elevated-default": {
82
+ "name": "background.elevated",
83
+ "description": "Background of popovers, sheets, modals, etc.",
84
+ "subState": "default",
85
+ "darkValue": "#24282d",
86
+ "darkOriginalValue": "night-black.dark01",
87
+ "lightValue": "#ffffff",
88
+ "lightOriginalValue": "neutral.white"
89
+ },
81
90
  "color-background-onAccent-default": {
82
91
  "name": "background.onAccent",
83
92
  "subState": "default",
@@ -187,7 +187,7 @@
187
187
  "value": "8px",
188
188
  "original": 8,
189
189
  "variableName": "$size-border-radius-s",
190
- "isUsed": false
190
+ "isUsed": true
191
191
  },
192
192
  {
193
193
  "category": "size",
@@ -177,6 +177,11 @@ type Variables = {
177
177
  "black": number;
178
178
  "inherit": string;
179
179
  };
180
+ "zIndex": {
181
+ "dropdown": number;
182
+ "tooltip": number;
183
+ "modal": number;
184
+ };
180
185
  };
181
186
  type AmbossTheme = {
182
187
  "color": {
@@ -196,6 +201,9 @@ type AmbossTheme = {
196
201
  "active": string;
197
202
  "disabled": string;
198
203
  };
204
+ "elevated": {
205
+ "default": string;
206
+ };
199
207
  "onAccent": {
200
208
  "default": string;
201
209
  "hover": string;
@@ -176,6 +176,11 @@ const variables = {
176
176
  "bold": 700,
177
177
  "black": 900,
178
178
  "inherit": "inherit"
179
+ },
180
+ "zIndex": {
181
+ "dropdown": 1,
182
+ "tooltip": 2,
183
+ "modal": 3
179
184
  }
180
185
  };
181
186
  const ambossVisualConfiguration = {
@@ -200,6 +205,9 @@ const ambossVisualConfiguration = {
200
205
  "active": "#233d3d",
201
206
  "disabled": "rgba(40, 129, 107, 0.3)"
202
207
  },
208
+ "elevated": {
209
+ "default": "#24282d"
210
+ },
203
211
  "onAccent": {
204
212
  "default": "#ffffff",
205
213
  "hover": "#ffffff",
@@ -820,6 +828,9 @@ const ambossVisualConfiguration = {
820
828
  "active": "#0a5c45",
821
829
  "disabled": "rgba(15, 169, 128, 0.3)"
822
830
  },
831
+ "elevated": {
832
+ "default": "#ffffff"
833
+ },
823
834
  "onAccent": {
824
835
  "default": "#ffffff",
825
836
  "hover": "#ffffff",