@amboss/design-system 1.16.6 → 1.17.0

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.
@@ -0,0 +1,124 @@
1
+ 'use strict';
2
+
3
+ var React = require('react');
4
+ var FocusTrap = require('focus-trap-react');
5
+ var TooltipContent = require('../Tooltip/TooltipContent.js');
6
+
7
+ function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
8
+
9
+ var React__default = /*#__PURE__*/_interopDefault(React);
10
+ var FocusTrap__default = /*#__PURE__*/_interopDefault(FocusTrap);
11
+
12
+ const FocusTrapContent = /*#__PURE__*/React__default.default.forwardRef((_ref, ref) => {
13
+ let {
14
+ children
15
+ } = _ref;
16
+ return /*#__PURE__*/React__default.default.createElement("div", {
17
+ ref: ref
18
+ }, children);
19
+ });
20
+ function Popover(_ref2) {
21
+ let {
22
+ placement = "auto",
23
+ content,
24
+ children,
25
+ contentPadding = "m",
26
+ externalTriggerRef,
27
+ portalContainer,
28
+ isVisible: isPopoverVisible,
29
+ "data-e2e-test-id": dataE2eTestId,
30
+ onVisibilityChange
31
+ } = _ref2;
32
+ const tooltipId = React.useMemo(() => `DSTooltip_${Math.floor(Date.now() * Math.random())}`, []);
33
+ const [isVisible, setVisible] = React.useState(isPopoverVisible);
34
+ const internalTriggerRef = React.useRef(null);
35
+ const triggerRef = externalTriggerRef || internalTriggerRef;
36
+ const isOutsideClickOnTrigger = React.useRef(false);
37
+ const toggleVisibility = React.useCallback(status => {
38
+ setVisible(status);
39
+ if (onVisibilityChange) {
40
+ onVisibilityChange(status);
41
+ }
42
+ }, [onVisibilityChange]);
43
+
44
+ // Outside click is also fired when the popover is open and trigger is clicked. `isOutsideClickOnTrigger` saves this condition and we check for it so as to not toggle the popover twice.
45
+ const handleTriggerClick = React.useCallback(() => {
46
+ if (!isOutsideClickOnTrigger.current) {
47
+ toggleVisibility(!isVisible);
48
+ } else {
49
+ // reset this value so that popover can open in next click
50
+ isOutsideClickOnTrigger.current = false;
51
+ }
52
+ }, [toggleVisibility, isVisible]);
53
+ const handleClickOutsideDeactivates = React.useCallback(evt => {
54
+ if (triggerRef.current.contains(evt.target)) {
55
+ isOutsideClickOnTrigger.current = true;
56
+ }
57
+ return true;
58
+ }, [triggerRef, isOutsideClickOnTrigger]);
59
+ React.useEffect(() => {
60
+ toggleVisibility(isPopoverVisible);
61
+ }, [isPopoverVisible, toggleVisibility]);
62
+ React.useEffect(() => {
63
+ let trigger;
64
+ if (externalTriggerRef?.current && !children) {
65
+ trigger = externalTriggerRef.current;
66
+ trigger.setAttribute("tabindex", "0");
67
+ trigger.addEventListener("click", handleTriggerClick);
68
+ }
69
+ return () => {
70
+ if (trigger) {
71
+ trigger.removeEventListener("click", handleTriggerClick);
72
+ }
73
+ };
74
+ }, [externalTriggerRef, children, handleTriggerClick]);
75
+ React.useEffect(() => {
76
+ if (externalTriggerRef?.current && !children) {
77
+ const trigger = externalTriggerRef.current;
78
+ if (isVisible) {
79
+ trigger.setAttribute("aria-expanded", true);
80
+ trigger.setAttribute("aria-controls", tooltipId);
81
+ } else {
82
+ trigger.removeAttribute("aria-expanded");
83
+ trigger.removeAttribute("aria-controls");
84
+ }
85
+ }
86
+ }, [externalTriggerRef, children, tooltipId, isVisible]);
87
+ const triggerElm = children ? /*#__PURE__*/React__default.default.cloneElement(children, {
88
+ ref: triggerRef,
89
+ ...(isVisible && {
90
+ "aria-expanded": true,
91
+ "aria-controls": tooltipId
92
+ }),
93
+ tabIndex: 0,
94
+ onClick: handleTriggerClick
95
+ }) : null;
96
+ const contentElm = /*#__PURE__*/React__default.default.createElement(FocusTrap__default.default, {
97
+ focusTrapOptions: {
98
+ clickOutsideDeactivates: handleClickOutsideDeactivates,
99
+ // de-active focus trap on outside click
100
+ escapeDeactivates: true,
101
+ // de-activate focus trap on escape key
102
+ fallbackFocus: `#${tooltipId}`,
103
+ // set focus to tooltip content container if it has no focusable element
104
+ onDeactivate: () => {
105
+ toggleVisibility(false);
106
+ }
107
+ }
108
+ }, /*#__PURE__*/React__default.default.createElement(FocusTrapContent, null, content));
109
+ const tooltipElm = /*#__PURE__*/React__default.default.createElement(TooltipContent.TooltipContent, {
110
+ dataDSId: "Popover",
111
+ content: contentElm,
112
+ contentPadding: contentPadding,
113
+ placement: placement,
114
+ portalContainer: portalContainer,
115
+ dataE2eTestId: dataE2eTestId,
116
+ isVisible: isVisible,
117
+ tooltipId: tooltipId,
118
+ tabIndex: -1,
119
+ triggerRef: triggerRef
120
+ });
121
+ return /*#__PURE__*/React__default.default.createElement(React__default.default.Fragment, null, triggerElm, tooltipElm);
122
+ }
123
+
124
+ exports.Popover = Popover;
@@ -1,113 +1,15 @@
1
1
  'use strict';
2
2
 
3
- var _styled = require('@emotion/styled/base');
4
3
  var React = require('react');
5
- var react = require('@emotion/react');
6
- var ReactDOM = require('react-dom');
7
- var useDocument = require('../../shared/useDocument.js');
8
- var useWindow = require('../../shared/useWindow.js');
9
- var SubThemeProvider = require('../SubThemeProvider/SubThemeProvider.js');
10
4
  var Text = require('../Typography/Text/Text.js');
11
- var _zindex = require('../../../build-tokens/_zindex.json.js');
12
- var utils = require('./utils.js');
5
+ var TooltipContent = require('./TooltipContent.js');
13
6
 
14
7
  function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
15
8
 
16
- var _styled__default = /*#__PURE__*/_interopDefault(_styled);
17
9
  var React__default = /*#__PURE__*/_interopDefault(React);
18
10
 
19
- const ANIMATION_DURATION = 200;
20
11
  const SHOW_HIDE_DELAY = 200;
21
- const StyledContainer = /*#__PURE__*/_styled__default.default("div", process.env.NODE_ENV === "production" ? {
22
- target: "e2kei841"
23
- } : {
24
- target: "e2kei841",
25
- label: "StyledContainer"
26
- })(_ref => {
27
- let {
28
- theme,
29
- horizontalPlacement,
30
- verticalPlacement
31
- } = _ref;
32
- const animationDistance = verticalPlacement === "top" ? `${utils.ANIMATION_DISTANCE}px` : `-${utils.ANIMATION_DISTANCE}px`;
33
- const animation = react.keyframes({
34
- to: {
35
- opacity: 1,
36
- transform: horizontalPlacement === "center" ? `translate(-50%, ${animationDistance})` : `translateY(${animationDistance})`
37
- }
38
- });
39
- return {
40
- position: "absolute",
41
- zIndex: _zindex.default.tooltip.value,
42
- opacity: 0,
43
- animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,
44
- borderRadius: theme.variables.size.borderRadius.xs,
45
- backgroundColor: theme.values.color.background.primary.default,
46
- maxWidth: "224px",
47
- boxSizing: "border-box",
48
- padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,
49
- ...(horizontalPlacement === "center" && {
50
- transform: "translate(-50%)"
51
- })
52
- };
53
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.tsx"],"names":[],"mappings":"AAyEwB","file":"Tooltip.tsx","sourcesContent":["import React, {\n  useState,\n  useRef,\n  useLayoutEffect,\n  useEffect,\n  useCallback,\n  MutableRefObject,\n  useMemo,\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 { Text } from \"../Typography/Text/Text\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ARROW_SIZE,\n  ARROW_OFFSET,\n  ANIMATION_DISTANCE,\n  getTooltipPosition,\n} from \"./utils\";\n\nexport type BaseProps = {\n  /* Tooltip content */\n  content: string;\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  \"data-e2e-test-id\"?: string;\n  /* Called when tooltip appears and disappears */\n  onVisibilityChange?: (isVisible: boolean) => void;\n};\n\nexport type ConditionalProps =\n  | {\n      externalTriggerRef?: never;\n      /* Trigger element - wrap trigger element within Tooltip - takes precedence over trigger prop */\n      children: React.ReactElement;\n    }\n  | {\n      children?: never;\n      /* Trigger element ref - pass in an external trigger element */\n      externalTriggerRef: MutableRefObject<any>;\n    };\n\nexport type TooltipProps = BaseProps & ConditionalProps;\n\nexport type TooltipPosition = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = {\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({ theme, horizontalPlacement, verticalPlacement }) => {\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    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      borderRadius: theme.variables.size.borderRadius.xs,\n      backgroundColor: theme.values.color.background.primary.default,\n      maxWidth: \"224px\",\n      boxSizing: \"border-box\",\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n};\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({ theme, verticalPlacement, horizontalPlacement }) => ({\n    position: \"absolute\",\n    width: 0,\n    height: 0,\n    borderLeft: `${ARROW_SIZE}px solid transparent`,\n    borderRight: `${ARROW_SIZE}px solid transparent`,\n\n    ...(verticalPlacement === \"top\" && {\n      top: \"100%\",\n      borderTop: `${ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`,\n    }),\n\n    ...(verticalPlacement === \"bottom\" && {\n      top: `-${ARROW_SIZE}px`,\n      borderBottom: `${ARROW_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: `${ARROW_OFFSET}px`,\n    }),\n\n    ...(horizontalPlacement === \"left\" && {\n      right: `${ARROW_OFFSET}px`,\n    }),\n  })\n);\n\nconst initialPosition: TooltipPosition = {\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\nexport function Tooltip({\n  placement = \"auto\",\n  content,\n  children,\n  externalTriggerRef,\n  portalContainer,\n  \"data-e2e-test-id\": dataE2eTestId,\n  onVisibilityChange,\n}: TooltipProps): React.ReactElement {\n  const tooltipId = useMemo(\n    () => `DSTooltip_${Math.floor(Date.now() * Math.random())}`,\n    []\n  );\n  const [position, setPosition] = useState(initialPosition);\n  const [isVisible, setVisible] = useState(false);\n  const triggeredByEvent = useRef(null); // indicates if triggered by hover or focus\n  const isTooltipHovered = useRef(false);\n  const isTriggerHovered = useRef(false);\n  const internalTriggerRef = useRef(null);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n  const hideTooltipTimeoutId = useRef(null);\n  const showTooltipTimeoutId = useRef(null);\n  const triggerRef = externalTriggerRef || internalTriggerRef;\n\n  const toggleVisibility = useCallback(\n    (status: boolean) => {\n      setVisible(status);\n\n      if (onVisibilityChange) {\n        onVisibilityChange(status);\n      }\n\n      // log time when tooltip closes\n      if (!status) {\n        lastTooltipHideTimestamp = Date.now();\n      }\n    },\n    [onVisibilityChange]\n  );\n\n  const handleTriggerPointerEnter = useCallback(() => {\n    isTriggerHovered.current = true;\n    if (!isTooltipHovered.current) {\n      clearTimeout(showTooltipTimeoutId.current);\n      // Delay show tooltip to prevent flickering when mouse moves quickly over trigger\n      showTooltipTimeoutId.current = setTimeout(() => {\n        if (isTriggerHovered.current) {\n          triggeredByEvent.current = \"hover\";\n          toggleVisibility(true);\n        }\n      }, SHOW_HIDE_DELAY);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerPointerLeave = useCallback(() => {\n    isTriggerHovered.current = false;\n    clearTimeout(hideTooltipTimeoutId.current);\n    // Delay removing tooltip from DOM to allow hover over tooltip element\n    hideTooltipTimeoutId.current = setTimeout(() => {\n      if (\n        !isTooltipHovered.current &&\n        triggeredByEvent.current === \"hover\" &&\n        !isTriggerHovered.current\n      ) {\n        toggleVisibility(false);\n      }\n    }, SHOW_HIDE_DELAY);\n  }, [toggleVisibility]);\n\n  const handleTriggerFocus = useCallback(() => {\n    triggeredByEvent.current = \"focus\";\n    toggleVisibility(true);\n  }, [toggleVisibility]);\n\n  const handleTriggerBlur = useCallback(() => {\n    if (triggeredByEvent.current === \"focus\") {\n      toggleVisibility(false);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerKeyDown = useCallback(\n    (evt) => {\n      if (evt.key === \"Escape\") {\n        toggleVisibility(false);\n      }\n    },\n    [toggleVisibility]\n  );\n\n  const handleTooltipPointerEnter = () => {\n    isTooltipHovered.current = true;\n  };\n\n  const handleTooltipPointerLeave = () => {\n    isTooltipHovered.current = false;\n    if (triggeredByEvent.current === \"hover\") {\n      toggleVisibility(false);\n    }\n  };\n\n  useEffect(\n    () => () => {\n      // clear timers\n      clearTimeout(showTooltipTimeoutId.current);\n      clearTimeout(hideTooltipTimeoutId.current);\n    },\n    []\n  );\n\n  useEffect(() => {\n    let trigger: HTMLElement;\n\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      trigger = externalTriggerRef.current;\n\n      trigger.setAttribute(\"tabindex\", \"0\");\n      trigger.addEventListener(\"pointerenter\", handleTriggerPointerEnter);\n      trigger.addEventListener(\"pointerleave\", handleTriggerPointerLeave);\n      trigger.addEventListener(\"focus\", handleTriggerFocus);\n      trigger.addEventListener(\"blur\", handleTriggerBlur);\n      trigger.addEventListener(\"keydown\", handleTriggerKeyDown);\n    }\n\n    return () => {\n      if (trigger) {\n        trigger.removeEventListener(\"pointerenter\", handleTriggerPointerEnter);\n        trigger.removeEventListener(\"pointerleave\", handleTriggerPointerLeave);\n        trigger.removeEventListener(\"focus\", handleTriggerFocus);\n        trigger.removeEventListener(\"blur\", handleTriggerBlur);\n        trigger.removeEventListener(\"keydown\", handleTriggerKeyDown);\n      }\n    };\n  }, [\n    externalTriggerRef,\n    children,\n    handleTriggerPointerEnter,\n    handleTriggerPointerLeave,\n    handleTriggerFocus,\n    handleTriggerBlur,\n    handleTriggerKeyDown,\n  ]);\n\n  useEffect(() => {\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      const trigger = externalTriggerRef.current;\n\n      if (isVisible) {\n        trigger.setAttribute(\"aria-describedby\", tooltipId);\n      } else {\n        trigger.removeAttribute(\"aria-describedby\");\n      }\n    }\n  }, [externalTriggerRef, children, tooltipId, isVisible]);\n\n  useLayoutEffect(() => {\n    if (isVisible && triggerRef.current && tooltipRef.current) {\n      // calculate tooltip position\n      setPosition(\n        getTooltipPosition(placement, triggerRef, tooltipRef, document, window)\n      );\n    }\n  }, [isVisible, placement, triggerRef, document, window]);\n\n  let portal;\n\n  if (isVisible) {\n    const tooltipElm = (\n      <SubThemeProvider name=\"inverted\">\n        <StyledContainer\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"Tooltip\"\n          style={{\n            top: position.top,\n            left: position.left,\n            animationDuration: getAnimationDuration(),\n          }}\n          ref={tooltipRef}\n          id={tooltipId}\n          role=\"tooltip\"\n          aria-hidden=\"true\"\n          horizontalPlacement={position.horizontalPlacement}\n          verticalPlacement={position.verticalPlacement}\n          onPointerEnter={handleTooltipPointerEnter}\n          onPointerLeave={handleTooltipPointerLeave}\n        >\n          <Text size=\"s\">{content}</Text>\n          <StyledArrow\n            horizontalPlacement={position.horizontalPlacement}\n            verticalPlacement={position.verticalPlacement}\n          />\n        </StyledContainer>\n      </SubThemeProvider>\n    );\n\n    portal = createPortal(tooltipElm, portalContainer || document.body);\n  }\n\n  const triggerElm = children\n    ? React.cloneElement(children, {\n        ref: triggerRef,\n        ...(isVisible && {\n          \"aria-describedby\": tooltipId,\n        }),\n        tabIndex: 0,\n        onPointerEnter: handleTriggerPointerEnter,\n        onPointerLeave: handleTriggerPointerLeave,\n        onFocus: handleTriggerFocus,\n        onBlur: handleTriggerBlur,\n        onKeyDown: handleTriggerKeyDown,\n      })\n    : null;\n\n  return (\n    <>\n      {triggerElm}\n      {portal}\n    </>\n  );\n}\n"]} */");
54
- const StyledArrow = /*#__PURE__*/_styled__default.default("div", process.env.NODE_ENV === "production" ? {
55
- target: "e2kei840"
56
- } : {
57
- target: "e2kei840",
58
- label: "StyledArrow"
59
- })(_ref2 => {
60
- let {
61
- theme,
62
- verticalPlacement,
63
- horizontalPlacement
64
- } = _ref2;
65
- return {
66
- position: "absolute",
67
- width: 0,
68
- height: 0,
69
- borderLeft: `${utils.ARROW_SIZE}px solid transparent`,
70
- borderRight: `${utils.ARROW_SIZE}px solid transparent`,
71
- ...(verticalPlacement === "top" && {
72
- top: "100%",
73
- borderTop: `${utils.ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`
74
- }),
75
- ...(verticalPlacement === "bottom" && {
76
- top: `-${utils.ARROW_SIZE}px`,
77
- borderBottom: `${utils.ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`
78
- }),
79
- ...(horizontalPlacement === "center" && {
80
- left: "50%",
81
- transform: "translate(-50%)"
82
- }),
83
- ...(horizontalPlacement === "right" && {
84
- left: `${utils.ARROW_OFFSET}px`
85
- }),
86
- ...(horizontalPlacement === "left" && {
87
- right: `${utils.ARROW_OFFSET}px`
88
- })
89
- };
90
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.tsx"],"names":[],"mappings":"AAgHoB","file":"Tooltip.tsx","sourcesContent":["import React, {\n  useState,\n  useRef,\n  useLayoutEffect,\n  useEffect,\n  useCallback,\n  MutableRefObject,\n  useMemo,\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 { Text } from \"../Typography/Text/Text\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ARROW_SIZE,\n  ARROW_OFFSET,\n  ANIMATION_DISTANCE,\n  getTooltipPosition,\n} from \"./utils\";\n\nexport type BaseProps = {\n  /* Tooltip content */\n  content: string;\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  \"data-e2e-test-id\"?: string;\n  /* Called when tooltip appears and disappears */\n  onVisibilityChange?: (isVisible: boolean) => void;\n};\n\nexport type ConditionalProps =\n  | {\n      externalTriggerRef?: never;\n      /* Trigger element - wrap trigger element within Tooltip - takes precedence over trigger prop */\n      children: React.ReactElement;\n    }\n  | {\n      children?: never;\n      /* Trigger element ref - pass in an external trigger element */\n      externalTriggerRef: MutableRefObject<any>;\n    };\n\nexport type TooltipProps = BaseProps & ConditionalProps;\n\nexport type TooltipPosition = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = {\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({ theme, horizontalPlacement, verticalPlacement }) => {\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    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      borderRadius: theme.variables.size.borderRadius.xs,\n      backgroundColor: theme.values.color.background.primary.default,\n      maxWidth: \"224px\",\n      boxSizing: \"border-box\",\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n};\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({ theme, verticalPlacement, horizontalPlacement }) => ({\n    position: \"absolute\",\n    width: 0,\n    height: 0,\n    borderLeft: `${ARROW_SIZE}px solid transparent`,\n    borderRight: `${ARROW_SIZE}px solid transparent`,\n\n    ...(verticalPlacement === \"top\" && {\n      top: \"100%\",\n      borderTop: `${ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`,\n    }),\n\n    ...(verticalPlacement === \"bottom\" && {\n      top: `-${ARROW_SIZE}px`,\n      borderBottom: `${ARROW_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: `${ARROW_OFFSET}px`,\n    }),\n\n    ...(horizontalPlacement === \"left\" && {\n      right: `${ARROW_OFFSET}px`,\n    }),\n  })\n);\n\nconst initialPosition: TooltipPosition = {\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\nexport function Tooltip({\n  placement = \"auto\",\n  content,\n  children,\n  externalTriggerRef,\n  portalContainer,\n  \"data-e2e-test-id\": dataE2eTestId,\n  onVisibilityChange,\n}: TooltipProps): React.ReactElement {\n  const tooltipId = useMemo(\n    () => `DSTooltip_${Math.floor(Date.now() * Math.random())}`,\n    []\n  );\n  const [position, setPosition] = useState(initialPosition);\n  const [isVisible, setVisible] = useState(false);\n  const triggeredByEvent = useRef(null); // indicates if triggered by hover or focus\n  const isTooltipHovered = useRef(false);\n  const isTriggerHovered = useRef(false);\n  const internalTriggerRef = useRef(null);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n  const hideTooltipTimeoutId = useRef(null);\n  const showTooltipTimeoutId = useRef(null);\n  const triggerRef = externalTriggerRef || internalTriggerRef;\n\n  const toggleVisibility = useCallback(\n    (status: boolean) => {\n      setVisible(status);\n\n      if (onVisibilityChange) {\n        onVisibilityChange(status);\n      }\n\n      // log time when tooltip closes\n      if (!status) {\n        lastTooltipHideTimestamp = Date.now();\n      }\n    },\n    [onVisibilityChange]\n  );\n\n  const handleTriggerPointerEnter = useCallback(() => {\n    isTriggerHovered.current = true;\n    if (!isTooltipHovered.current) {\n      clearTimeout(showTooltipTimeoutId.current);\n      // Delay show tooltip to prevent flickering when mouse moves quickly over trigger\n      showTooltipTimeoutId.current = setTimeout(() => {\n        if (isTriggerHovered.current) {\n          triggeredByEvent.current = \"hover\";\n          toggleVisibility(true);\n        }\n      }, SHOW_HIDE_DELAY);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerPointerLeave = useCallback(() => {\n    isTriggerHovered.current = false;\n    clearTimeout(hideTooltipTimeoutId.current);\n    // Delay removing tooltip from DOM to allow hover over tooltip element\n    hideTooltipTimeoutId.current = setTimeout(() => {\n      if (\n        !isTooltipHovered.current &&\n        triggeredByEvent.current === \"hover\" &&\n        !isTriggerHovered.current\n      ) {\n        toggleVisibility(false);\n      }\n    }, SHOW_HIDE_DELAY);\n  }, [toggleVisibility]);\n\n  const handleTriggerFocus = useCallback(() => {\n    triggeredByEvent.current = \"focus\";\n    toggleVisibility(true);\n  }, [toggleVisibility]);\n\n  const handleTriggerBlur = useCallback(() => {\n    if (triggeredByEvent.current === \"focus\") {\n      toggleVisibility(false);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerKeyDown = useCallback(\n    (evt) => {\n      if (evt.key === \"Escape\") {\n        toggleVisibility(false);\n      }\n    },\n    [toggleVisibility]\n  );\n\n  const handleTooltipPointerEnter = () => {\n    isTooltipHovered.current = true;\n  };\n\n  const handleTooltipPointerLeave = () => {\n    isTooltipHovered.current = false;\n    if (triggeredByEvent.current === \"hover\") {\n      toggleVisibility(false);\n    }\n  };\n\n  useEffect(\n    () => () => {\n      // clear timers\n      clearTimeout(showTooltipTimeoutId.current);\n      clearTimeout(hideTooltipTimeoutId.current);\n    },\n    []\n  );\n\n  useEffect(() => {\n    let trigger: HTMLElement;\n\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      trigger = externalTriggerRef.current;\n\n      trigger.setAttribute(\"tabindex\", \"0\");\n      trigger.addEventListener(\"pointerenter\", handleTriggerPointerEnter);\n      trigger.addEventListener(\"pointerleave\", handleTriggerPointerLeave);\n      trigger.addEventListener(\"focus\", handleTriggerFocus);\n      trigger.addEventListener(\"blur\", handleTriggerBlur);\n      trigger.addEventListener(\"keydown\", handleTriggerKeyDown);\n    }\n\n    return () => {\n      if (trigger) {\n        trigger.removeEventListener(\"pointerenter\", handleTriggerPointerEnter);\n        trigger.removeEventListener(\"pointerleave\", handleTriggerPointerLeave);\n        trigger.removeEventListener(\"focus\", handleTriggerFocus);\n        trigger.removeEventListener(\"blur\", handleTriggerBlur);\n        trigger.removeEventListener(\"keydown\", handleTriggerKeyDown);\n      }\n    };\n  }, [\n    externalTriggerRef,\n    children,\n    handleTriggerPointerEnter,\n    handleTriggerPointerLeave,\n    handleTriggerFocus,\n    handleTriggerBlur,\n    handleTriggerKeyDown,\n  ]);\n\n  useEffect(() => {\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      const trigger = externalTriggerRef.current;\n\n      if (isVisible) {\n        trigger.setAttribute(\"aria-describedby\", tooltipId);\n      } else {\n        trigger.removeAttribute(\"aria-describedby\");\n      }\n    }\n  }, [externalTriggerRef, children, tooltipId, isVisible]);\n\n  useLayoutEffect(() => {\n    if (isVisible && triggerRef.current && tooltipRef.current) {\n      // calculate tooltip position\n      setPosition(\n        getTooltipPosition(placement, triggerRef, tooltipRef, document, window)\n      );\n    }\n  }, [isVisible, placement, triggerRef, document, window]);\n\n  let portal;\n\n  if (isVisible) {\n    const tooltipElm = (\n      <SubThemeProvider name=\"inverted\">\n        <StyledContainer\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"Tooltip\"\n          style={{\n            top: position.top,\n            left: position.left,\n            animationDuration: getAnimationDuration(),\n          }}\n          ref={tooltipRef}\n          id={tooltipId}\n          role=\"tooltip\"\n          aria-hidden=\"true\"\n          horizontalPlacement={position.horizontalPlacement}\n          verticalPlacement={position.verticalPlacement}\n          onPointerEnter={handleTooltipPointerEnter}\n          onPointerLeave={handleTooltipPointerLeave}\n        >\n          <Text size=\"s\">{content}</Text>\n          <StyledArrow\n            horizontalPlacement={position.horizontalPlacement}\n            verticalPlacement={position.verticalPlacement}\n          />\n        </StyledContainer>\n      </SubThemeProvider>\n    );\n\n    portal = createPortal(tooltipElm, portalContainer || document.body);\n  }\n\n  const triggerElm = children\n    ? React.cloneElement(children, {\n        ref: triggerRef,\n        ...(isVisible && {\n          \"aria-describedby\": tooltipId,\n        }),\n        tabIndex: 0,\n        onPointerEnter: handleTriggerPointerEnter,\n        onPointerLeave: handleTriggerPointerLeave,\n        onFocus: handleTriggerFocus,\n        onBlur: handleTriggerBlur,\n        onKeyDown: handleTriggerKeyDown,\n      })\n    : null;\n\n  return (\n    <>\n      {triggerElm}\n      {portal}\n    </>\n  );\n}\n"]} */");
91
- const initialPosition = {
92
- top: 0,
93
- left: 0,
94
- verticalPlacement: "top",
95
- horizontalPlacement: "center"
96
- };
97
- let lastTooltipHideTimestamp = 0;
98
-
99
- /* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */
100
- function getAnimationDuration() {
101
- let animationDuration = `${ANIMATION_DURATION}ms`;
102
- if (lastTooltipHideTimestamp) {
103
- const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;
104
- if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {
105
- animationDuration = "0ms";
106
- }
107
- }
108
- return animationDuration;
109
- }
110
- function Tooltip(_ref3) {
12
+ function Tooltip(_ref) {
111
13
  let {
112
14
  placement = "auto",
113
15
  content,
@@ -116,30 +18,21 @@ function Tooltip(_ref3) {
116
18
  portalContainer,
117
19
  "data-e2e-test-id": dataE2eTestId,
118
20
  onVisibilityChange
119
- } = _ref3;
21
+ } = _ref;
120
22
  const tooltipId = React.useMemo(() => `DSTooltip_${Math.floor(Date.now() * Math.random())}`, []);
121
- const [position, setPosition] = React.useState(initialPosition);
122
23
  const [isVisible, setVisible] = React.useState(false);
123
24
  const triggeredByEvent = React.useRef(null); // indicates if triggered by hover or focus
124
25
  const isTooltipHovered = React.useRef(false);
125
26
  const isTriggerHovered = React.useRef(false);
126
- const internalTriggerRef = React.useRef(null);
127
- const tooltipRef = React.useRef(null);
128
- const document = useDocument.useDocument();
129
- const window = useWindow.useWindow();
130
27
  const hideTooltipTimeoutId = React.useRef(null);
131
28
  const showTooltipTimeoutId = React.useRef(null);
29
+ const internalTriggerRef = React.useRef(null);
132
30
  const triggerRef = externalTriggerRef || internalTriggerRef;
133
31
  const toggleVisibility = React.useCallback(status => {
134
32
  setVisible(status);
135
33
  if (onVisibilityChange) {
136
34
  onVisibilityChange(status);
137
35
  }
138
-
139
- // log time when tooltip closes
140
- if (!status) {
141
- lastTooltipHideTimestamp = Date.now();
142
- }
143
36
  }, [onVisibilityChange]);
144
37
  const handleTriggerPointerEnter = React.useCallback(() => {
145
38
  isTriggerHovered.current = true;
@@ -194,7 +87,7 @@ function Tooltip(_ref3) {
194
87
  }, []);
195
88
  React.useEffect(() => {
196
89
  let trigger;
197
- if (externalTriggerRef && externalTriggerRef.current && !children) {
90
+ if (externalTriggerRef?.current && !children) {
198
91
  trigger = externalTriggerRef.current;
199
92
  trigger.setAttribute("tabindex", "0");
200
93
  trigger.addEventListener("pointerenter", handleTriggerPointerEnter);
@@ -214,7 +107,7 @@ function Tooltip(_ref3) {
214
107
  };
215
108
  }, [externalTriggerRef, children, handleTriggerPointerEnter, handleTriggerPointerLeave, handleTriggerFocus, handleTriggerBlur, handleTriggerKeyDown]);
216
109
  React.useEffect(() => {
217
- if (externalTriggerRef && externalTriggerRef.current && !children) {
110
+ if (externalTriggerRef?.current && !children) {
218
111
  const trigger = externalTriggerRef.current;
219
112
  if (isVisible) {
220
113
  trigger.setAttribute("aria-describedby", tooltipId);
@@ -223,40 +116,6 @@ function Tooltip(_ref3) {
223
116
  }
224
117
  }
225
118
  }, [externalTriggerRef, children, tooltipId, isVisible]);
226
- React.useLayoutEffect(() => {
227
- if (isVisible && triggerRef.current && tooltipRef.current) {
228
- // calculate tooltip position
229
- setPosition(utils.getTooltipPosition(placement, triggerRef, tooltipRef, document, window));
230
- }
231
- }, [isVisible, placement, triggerRef, document, window]);
232
- let portal;
233
- if (isVisible) {
234
- const tooltipElm = /*#__PURE__*/React__default.default.createElement(SubThemeProvider.SubThemeProvider, {
235
- name: "inverted"
236
- }, /*#__PURE__*/React__default.default.createElement(StyledContainer, {
237
- "data-e2e-test-id": dataE2eTestId,
238
- "data-ds-id": "Tooltip",
239
- style: {
240
- top: position.top,
241
- left: position.left,
242
- animationDuration: getAnimationDuration()
243
- },
244
- ref: tooltipRef,
245
- id: tooltipId,
246
- role: "tooltip",
247
- "aria-hidden": "true",
248
- horizontalPlacement: position.horizontalPlacement,
249
- verticalPlacement: position.verticalPlacement,
250
- onPointerEnter: handleTooltipPointerEnter,
251
- onPointerLeave: handleTooltipPointerLeave
252
- }, /*#__PURE__*/React__default.default.createElement(Text.Text, {
253
- size: "s"
254
- }, content), /*#__PURE__*/React__default.default.createElement(StyledArrow, {
255
- horizontalPlacement: position.horizontalPlacement,
256
- verticalPlacement: position.verticalPlacement
257
- })));
258
- portal = /*#__PURE__*/ReactDOM.createPortal(tooltipElm, portalContainer || document.body);
259
- }
260
119
  const triggerElm = children ? /*#__PURE__*/React__default.default.cloneElement(children, {
261
120
  ref: triggerRef,
262
121
  ...(isVisible && {
@@ -269,7 +128,24 @@ function Tooltip(_ref3) {
269
128
  onBlur: handleTriggerBlur,
270
129
  onKeyDown: handleTriggerKeyDown
271
130
  }) : null;
272
- return /*#__PURE__*/React__default.default.createElement(React__default.default.Fragment, null, triggerElm, portal);
131
+ const contentElm = /*#__PURE__*/React__default.default.createElement(Text.Text, {
132
+ size: "s"
133
+ }, content);
134
+ const tooltipElm = /*#__PURE__*/React__default.default.createElement(TooltipContent.TooltipContent, {
135
+ dataDSId: "Tooltip",
136
+ content: contentElm,
137
+ placement: placement,
138
+ portalContainer: portalContainer,
139
+ dataE2eTestId: dataE2eTestId,
140
+ isVisible: isVisible,
141
+ tooltipId: tooltipId,
142
+ triggerRef: triggerRef,
143
+ "aria-hidden": true,
144
+ role: "tooltip",
145
+ onTooltipPointerEnter: handleTooltipPointerEnter,
146
+ onTooltipPointerLeave: handleTooltipPointerLeave
147
+ });
148
+ return /*#__PURE__*/React__default.default.createElement(React__default.default.Fragment, null, triggerElm, tooltipElm);
273
149
  }
274
150
 
275
151
  exports.Tooltip = Tooltip;