@amboss/design-system 1.13.5 → 1.13.6

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 (34) hide show
  1. package/build/cjs/build-tokens/_zindex.json.js +10 -0
  2. package/build/cjs/build-tokens/visualConfig.js +142 -0
  3. package/build/cjs/scss/themes/dark.scss +1 -1
  4. package/build/cjs/scss/themes/light.scss +1 -1
  5. package/build/cjs/src/components/Button/Button.js +18 -32
  6. package/build/cjs/src/components/Link/Link.js +5 -4
  7. package/build/cjs/src/components/Tooltip/Tooltip.js +275 -0
  8. package/build/cjs/src/components/Tooltip/utils.js +81 -0
  9. package/build/cjs/src/index.js +2 -0
  10. package/build/esm/build-tokens/_subThemeType.d.ts +1 -1
  11. package/build/esm/build-tokens/_zindex.json.js +4 -0
  12. package/build/esm/build-tokens/_zindex.json.js.map +1 -0
  13. package/build/esm/build-tokens/visualConfig.d.ts +71 -0
  14. package/build/esm/build-tokens/visualConfig.js +142 -0
  15. package/build/esm/build-tokens/visualConfig.js.map +1 -1
  16. package/build/esm/scss/themes/dark.scss +1 -1
  17. package/build/esm/scss/themes/light.scss +1 -1
  18. package/build/esm/src/components/Button/Button.d.ts +0 -1
  19. package/build/esm/src/components/Button/Button.js +18 -32
  20. package/build/esm/src/components/Button/Button.js.map +1 -1
  21. package/build/esm/src/components/Link/Link.d.ts +4 -5
  22. package/build/esm/src/components/Link/Link.js +5 -4
  23. package/build/esm/src/components/Link/Link.js.map +1 -1
  24. package/build/esm/src/components/Tooltip/Tooltip.d.ts +23 -0
  25. package/build/esm/src/components/Tooltip/Tooltip.js +269 -0
  26. package/build/esm/src/components/Tooltip/Tooltip.js.map +1 -0
  27. package/build/esm/src/components/Tooltip/utils.d.ts +11 -0
  28. package/build/esm/src/components/Tooltip/utils.js +75 -0
  29. package/build/esm/src/components/Tooltip/utils.js.map +1 -0
  30. package/build/esm/src/index.d.ts +1 -0
  31. package/build/esm/src/index.js +1 -0
  32. package/build/esm/src/index.js.map +1 -1
  33. package/package.json +2 -1
  34. package/build/esm/src/components/Button/mock.d.ts +0 -326
@@ -0,0 +1,275 @@
1
+ 'use strict';
2
+
3
+ var _styled = require('@emotion/styled/base');
4
+ 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
+ var Text = require('../Typography/Text/Text.js');
11
+ var _zindex = require('../../../build-tokens/_zindex.json.js');
12
+ var utils = require('./utils.js');
13
+
14
+ function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
15
+
16
+ var _styled__default = /*#__PURE__*/_interopDefault(_styled);
17
+ var React__default = /*#__PURE__*/_interopDefault(React);
18
+
19
+ const ANIMATION_DURATION = 200;
20
+ 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) {
111
+ let {
112
+ placement = "auto",
113
+ content,
114
+ children,
115
+ externalTriggerRef,
116
+ portalContainer,
117
+ "data-e2e-test-id": dataE2eTestId,
118
+ onVisibilityChange
119
+ } = _ref3;
120
+ const tooltipId = React.useMemo(() => `DSTooltip_${Math.floor(Date.now() * Math.random())}`, []);
121
+ const [position, setPosition] = React.useState(initialPosition);
122
+ const [isVisible, setVisible] = React.useState(false);
123
+ const triggeredByEvent = React.useRef(null); // indicates if triggered by hover or focus
124
+ const isTooltipHovered = React.useRef(false);
125
+ 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
+ const hideTooltipTimeoutId = React.useRef(null);
131
+ const showTooltipTimeoutId = React.useRef(null);
132
+ const triggerRef = externalTriggerRef || internalTriggerRef;
133
+ const toggleVisibility = React.useCallback(status => {
134
+ setVisible(status);
135
+ if (onVisibilityChange) {
136
+ onVisibilityChange(status);
137
+ }
138
+
139
+ // log time when tooltip closes
140
+ if (!status) {
141
+ lastTooltipHideTimestamp = Date.now();
142
+ }
143
+ }, [onVisibilityChange]);
144
+ const handleTriggerPointerEnter = React.useCallback(() => {
145
+ isTriggerHovered.current = true;
146
+ if (!isTooltipHovered.current) {
147
+ clearTimeout(showTooltipTimeoutId.current);
148
+ // Delay show tooltip to prevent flickering when mouse moves quickly over trigger
149
+ showTooltipTimeoutId.current = setTimeout(() => {
150
+ if (isTriggerHovered.current) {
151
+ triggeredByEvent.current = "hover";
152
+ toggleVisibility(true);
153
+ }
154
+ }, SHOW_HIDE_DELAY);
155
+ }
156
+ }, [toggleVisibility]);
157
+ const handleTriggerPointerLeave = React.useCallback(() => {
158
+ isTriggerHovered.current = false;
159
+ clearTimeout(hideTooltipTimeoutId.current);
160
+ // Delay removing tooltip from DOM to allow hover over tooltip element
161
+ hideTooltipTimeoutId.current = setTimeout(() => {
162
+ if (!isTooltipHovered.current && triggeredByEvent.current === "hover" && !isTriggerHovered.current) {
163
+ toggleVisibility(false);
164
+ }
165
+ }, SHOW_HIDE_DELAY);
166
+ }, [toggleVisibility]);
167
+ const handleTriggerFocus = React.useCallback(() => {
168
+ triggeredByEvent.current = "focus";
169
+ toggleVisibility(true);
170
+ }, [toggleVisibility]);
171
+ const handleTriggerBlur = React.useCallback(() => {
172
+ if (triggeredByEvent.current === "focus") {
173
+ toggleVisibility(false);
174
+ }
175
+ }, [toggleVisibility]);
176
+ const handleTriggerKeyDown = React.useCallback(evt => {
177
+ if (evt.key === "Escape") {
178
+ toggleVisibility(false);
179
+ }
180
+ }, [toggleVisibility]);
181
+ const handleTooltipPointerEnter = () => {
182
+ isTooltipHovered.current = true;
183
+ };
184
+ const handleTooltipPointerLeave = () => {
185
+ isTooltipHovered.current = false;
186
+ if (triggeredByEvent.current === "hover") {
187
+ toggleVisibility(false);
188
+ }
189
+ };
190
+ React.useEffect(() => () => {
191
+ // clear timers
192
+ clearTimeout(showTooltipTimeoutId.current);
193
+ clearTimeout(hideTooltipTimeoutId.current);
194
+ }, []);
195
+ React.useEffect(() => {
196
+ let trigger;
197
+ if (externalTriggerRef && externalTriggerRef.current && !children) {
198
+ trigger = externalTriggerRef.current;
199
+ trigger.setAttribute("tabindex", "0");
200
+ trigger.addEventListener("pointerenter", handleTriggerPointerEnter);
201
+ trigger.addEventListener("pointerleave", handleTriggerPointerLeave);
202
+ trigger.addEventListener("focus", handleTriggerFocus);
203
+ trigger.addEventListener("blur", handleTriggerBlur);
204
+ trigger.addEventListener("keydown", handleTriggerKeyDown);
205
+ }
206
+ return () => {
207
+ if (trigger) {
208
+ trigger.removeEventListener("pointerenter", handleTriggerPointerEnter);
209
+ trigger.removeEventListener("pointerleave", handleTriggerPointerLeave);
210
+ trigger.removeEventListener("focus", handleTriggerFocus);
211
+ trigger.removeEventListener("blur", handleTriggerBlur);
212
+ trigger.removeEventListener("keydown", handleTriggerKeyDown);
213
+ }
214
+ };
215
+ }, [externalTriggerRef, children, handleTriggerPointerEnter, handleTriggerPointerLeave, handleTriggerFocus, handleTriggerBlur, handleTriggerKeyDown]);
216
+ React.useEffect(() => {
217
+ if (externalTriggerRef && externalTriggerRef.current && !children) {
218
+ const trigger = externalTriggerRef.current;
219
+ if (isVisible) {
220
+ trigger.setAttribute("aria-describedby", tooltipId);
221
+ } else {
222
+ trigger.removeAttribute("aria-describedby");
223
+ }
224
+ }
225
+ }, [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
+ const triggerElm = children ? /*#__PURE__*/React__default.default.cloneElement(children, {
261
+ ref: triggerRef,
262
+ ...(isVisible && {
263
+ "aria-describedby": tooltipId
264
+ }),
265
+ tabIndex: 0,
266
+ onPointerEnter: handleTriggerPointerEnter,
267
+ onPointerLeave: handleTriggerPointerLeave,
268
+ onFocus: handleTriggerFocus,
269
+ onBlur: handleTriggerBlur,
270
+ onKeyDown: handleTriggerKeyDown
271
+ }) : null;
272
+ return /*#__PURE__*/React__default.default.createElement(React__default.default.Fragment, null, triggerElm, portal);
273
+ }
274
+
275
+ exports.Tooltip = Tooltip;
@@ -0,0 +1,81 @@
1
+ 'use strict';
2
+
3
+ const DISTANCE_FROM_TRIGGER = 4;
4
+ const ANIMATION_DISTANCE = 8;
5
+ const ARROW_SIZE = 6;
6
+ const TOOLTIP_MARGIN = ARROW_SIZE + ANIMATION_DISTANCE + DISTANCE_FROM_TRIGGER;
7
+ const ARROW_OFFSET = 8;
8
+
9
+ /**
10
+ * Get tooltip position
11
+ */
12
+ function getTooltipPosition(placement, triggerRef, tooltipRef, document, window) {
13
+ const triggerRect = triggerRef.current.getBoundingClientRect();
14
+ const tooltipRect = tooltipRef.current.getBoundingClientRect();
15
+ const viewportWidth = document.documentElement.clientWidth;
16
+ let top = 0;
17
+ let left = 0;
18
+ let verticalPlacement = "top";
19
+ let horizontalPlacement = "center";
20
+ switch (placement) {
21
+ case "top":
22
+ case "bottom":
23
+ verticalPlacement = placement;
24
+ break;
25
+ case "top-left":
26
+ case "top-right":
27
+ case "bottom-left":
28
+ case "bottom-right":
29
+ {
30
+ const placements = placement.split("-");
31
+ verticalPlacement = placements[0];
32
+ horizontalPlacement = placements[1];
33
+ break;
34
+ }
35
+ default:
36
+ // If there is no space on the top of trigger, then place it at the bottom
37
+ if (triggerRect.top < tooltipRect.height) {
38
+ verticalPlacement = "bottom";
39
+ }
40
+
41
+ // Check if we have space on the left for half of the tooltip width
42
+ if (triggerRect.left >= tooltipRect.width / 2) {
43
+ // Check if we have space on the right for half of the tooltip width
44
+ if (triggerRect.left + triggerRect.width / 2 + tooltipRect.width / 2 > viewportWidth) {
45
+ // no space on the right
46
+ horizontalPlacement = "left";
47
+ }
48
+ } else {
49
+ // We don't have space on the left
50
+ horizontalPlacement = "right";
51
+ }
52
+ }
53
+ if (verticalPlacement === "top") {
54
+ top = triggerRect.top - tooltipRect.height - TOOLTIP_MARGIN;
55
+ } else {
56
+ top = triggerRect.bottom + TOOLTIP_MARGIN;
57
+ }
58
+ switch (horizontalPlacement) {
59
+ case "left":
60
+ left = triggerRect.left + triggerRect.width / 2 - tooltipRect.width + ARROW_OFFSET + ARROW_SIZE;
61
+ break;
62
+ case "right":
63
+ left = triggerRect.left + triggerRect.width / 2 - ARROW_OFFSET - ARROW_SIZE;
64
+ break;
65
+ default:
66
+ left = triggerRect.left + triggerRect.width / 2;
67
+ }
68
+ return {
69
+ top: top + window.scrollY,
70
+ left: left + window.scrollX,
71
+ horizontalPlacement,
72
+ verticalPlacement
73
+ };
74
+ }
75
+
76
+ exports.ANIMATION_DISTANCE = ANIMATION_DISTANCE;
77
+ exports.ARROW_OFFSET = ARROW_OFFSET;
78
+ exports.ARROW_SIZE = ARROW_SIZE;
79
+ exports.DISTANCE_FROM_TRIGGER = DISTANCE_FROM_TRIGGER;
80
+ exports.TOOLTIP_MARGIN = TOOLTIP_MARGIN;
81
+ exports.getTooltipPosition = getTooltipPosition;
@@ -46,6 +46,7 @@ var Collapsible = require('./components/Collapsible/Collapsible.js');
46
46
  var LoadingSpinner = require('./components/LoadingSpinner/LoadingSpinner.js');
47
47
  var Pagination = require('./components/Pagination/Pagination.js');
48
48
  var SegmentedControl = require('./components/Form/SegmentedControl/SegmentedControl.js');
49
+ var Tooltip = require('./components/Tooltip/Tooltip.js');
49
50
  var Input = require('./components/Form/Input/Input.js');
50
51
  var PasswordInput = require('./components/Form/PasswordInput/PasswordInput.js');
51
52
  var DataTable = require('./components/DataTable/DataTable.js');
@@ -123,6 +124,7 @@ exports.CollapsibleHeader = Collapsible.CollapsibleHeader;
123
124
  exports.LoadingSpinner = LoadingSpinner.LoadingSpinner;
124
125
  exports.Pagination = Pagination.Pagination;
125
126
  exports.SegmentedControl = SegmentedControl.SegmentedControl;
127
+ exports.Tooltip = Tooltip.Tooltip;
126
128
  exports.Input = Input.Input;
127
129
  exports.PasswordInput = PasswordInput.PasswordInput;
128
130
  exports.BaseDataTable = DataTable.BaseDataTable;
@@ -1 +1 @@
1
- export declare type SubThemeTypes = 'dimmed' | 'error' | 'info' | 'success';
1
+ export declare type SubThemeTypes = 'dimmed' | 'error' | 'info' | 'inverted' | 'success';
@@ -0,0 +1,4 @@
1
+ var dropdown={category:"zIndex",value:1,variableName:"$z-index-dropdown"};var tooltip={category:"zIndex",value:2,variableName:"$z-index-tooltip"};var modal={category:"zIndex",value:3,variableName:"$z-index-modal"};var zIndices = {dropdown:dropdown,tooltip:tooltip,modal:modal};
2
+
3
+ export { zIndices as default, dropdown, modal, tooltip };
4
+ //# sourceMappingURL=_zindex.json.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"_zindex.json.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;"}
@@ -762,6 +762,77 @@ declare type AmbossTheme = {
762
762
  };
763
763
  };
764
764
  };
765
+ "inverted": {
766
+ "color": {
767
+ "background": {
768
+ "primary": {
769
+ "default": string;
770
+ };
771
+ "secondary": {
772
+ "default": string;
773
+ "hover": string;
774
+ };
775
+ "accent": {
776
+ "default": string;
777
+ "hover": string;
778
+ "active": string;
779
+ "disabled": string;
780
+ };
781
+ "transparent": {
782
+ "hover": string;
783
+ };
784
+ };
785
+ "text": {
786
+ "primary": {
787
+ "default": string;
788
+ "hover": string;
789
+ };
790
+ "secondary": {
791
+ "default": string;
792
+ "hover": string;
793
+ };
794
+ "tertiary": {
795
+ "default": string;
796
+ "hover": string;
797
+ };
798
+ "accent": {
799
+ "default": string;
800
+ "hover": string;
801
+ };
802
+ "onAccent": {
803
+ "default": string;
804
+ };
805
+ "error": {
806
+ "default": string;
807
+ };
808
+ "info": {
809
+ "default": string;
810
+ };
811
+ };
812
+ "icon": {
813
+ "primary": string;
814
+ "secondary": string;
815
+ "tertiary": string;
816
+ "accent": string;
817
+ "onAccent": string;
818
+ "info": string;
819
+ "error": string;
820
+ };
821
+ "border": {
822
+ "primary": {
823
+ "default": string;
824
+ "hover": string;
825
+ "active": string;
826
+ };
827
+ "accent": {
828
+ "default": string;
829
+ };
830
+ };
831
+ "divider": {
832
+ "primary": string;
833
+ };
834
+ };
835
+ };
765
836
  "success": {
766
837
  "color": {
767
838
  "background": {