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