@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.
- package/build/cjs/src/components/Popover/Popover.js +124 -0
- package/build/cjs/src/components/Tooltip/Tooltip.js +24 -148
- package/build/cjs/src/components/Tooltip/TooltipContent.js +210 -0
- package/build/cjs/src/components/Tooltip/utils.js +3 -3
- package/build/cjs/src/index.js +2 -0
- package/build/esm/src/components/Popover/Popover.d.ts +18 -0
- package/build/esm/src/components/Popover/Popover.js +118 -0
- package/build/esm/src/components/Popover/Popover.js.map +1 -0
- package/build/esm/src/components/Tooltip/Tooltip.d.ts +4 -10
- package/build/esm/src/components/Tooltip/Tooltip.js +25 -148
- package/build/esm/src/components/Tooltip/Tooltip.js.map +1 -1
- package/build/esm/src/components/Tooltip/TooltipContent.d.ts +25 -0
- package/build/esm/src/components/Tooltip/TooltipContent.js +204 -0
- package/build/esm/src/components/Tooltip/TooltipContent.js.map +1 -0
- package/build/esm/src/components/Tooltip/utils.d.ts +4 -3
- package/build/esm/src/components/Tooltip/utils.js +3 -3
- package/build/esm/src/components/Tooltip/utils.js.map +1 -1
- package/build/esm/src/index.d.ts +1 -0
- package/build/esm/src/index.js +1 -0
- package/build/esm/src/index.js.map +1 -1
- package/package.json +2 -1
|
@@ -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
|
|
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
|
-
|
|
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
|
-
} =
|
|
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
|
|
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
|
|
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
|
-
|
|
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;
|