@amboss/design-system 1.16.6 → 1.17.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 +75 -63
- package/build/cjs/src/components/Form/ToggleButton/ToggleButton.js +8 -8
- package/build/cjs/src/components/Popover/Popover.js +124 -0
- package/build/cjs/src/components/Tabs/Tabs.js +3 -3
- package/build/cjs/src/components/Tooltip/Tooltip.js +24 -148
- package/build/cjs/src/components/Tooltip/TooltipContent.js +210 -0
- package/build/cjs/src/components/Tooltip/utils.js +3 -3
- package/build/cjs/src/index.js +2 -0
- package/build/esm/build-tokens/_colors.json +74 -38
- package/build/esm/build-tokens/visualConfig.d.ts +9 -3
- package/build/esm/build-tokens/visualConfig.js +75 -63
- package/build/esm/build-tokens/visualConfig.js.map +1 -1
- package/build/esm/src/components/Form/ToggleButton/ToggleButton.js +8 -8
- package/build/esm/src/components/Form/ToggleButton/ToggleButton.js.map +1 -1
- package/build/esm/src/components/Popover/Popover.d.ts +18 -0
- package/build/esm/src/components/Popover/Popover.js +118 -0
- package/build/esm/src/components/Popover/Popover.js.map +1 -0
- package/build/esm/src/components/Tabs/Tabs.js +3 -3
- package/build/esm/src/components/Tabs/Tabs.js.map +1 -1
- package/build/esm/src/components/Tooltip/Tooltip.d.ts +4 -10
- package/build/esm/src/components/Tooltip/Tooltip.js +25 -148
- package/build/esm/src/components/Tooltip/Tooltip.js.map +1 -1
- package/build/esm/src/components/Tooltip/TooltipContent.d.ts +25 -0
- package/build/esm/src/components/Tooltip/TooltipContent.js +204 -0
- package/build/esm/src/components/Tooltip/TooltipContent.js.map +1 -0
- package/build/esm/src/components/Tooltip/utils.d.ts +4 -3
- package/build/esm/src/components/Tooltip/utils.js +3 -3
- package/build/esm/src/components/Tooltip/utils.js.map +1 -1
- package/build/esm/src/index.d.ts +1 -0
- package/build/esm/src/index.js +1 -0
- package/build/esm/src/index.js.map +1 -1
- package/build/scss/_theming.scss +19 -11
- package/build/scss/_variables.scss +6 -2
- package/package.json +2 -1
|
@@ -44,7 +44,7 @@ const StyledContainer = /*#__PURE__*/_styled__default.default("div", process.env
|
|
|
44
44
|
display: "none" // Chrome
|
|
45
45
|
}
|
|
46
46
|
};
|
|
47
|
-
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlRhYnMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQVl3QiIsImZpbGUiOiJUYWJzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIHJlYWN0L2pzeC1wcm9wcy1uby1zcHJlYWRpbmcgKi9cblxuaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QgZnJvbSBcInJlYWN0XCI7XG5cbmltcG9ydCB7IG1xIH0gZnJvbSBcIi4uLy4uL3NoYXJlZC9tZWRpYVF1ZXJpZXNcIjtcbmltcG9ydCB7IE1RLCBTcGFjZVNpemVzIH0gZnJvbSBcIi4uLy4uL3R5cGVzXCI7XG5pbXBvcnQgeyBJY29uLCBJY29uTmFtZSB9IGZyb20gXCIuLi9JY29uL0ljb25cIjtcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSBcIi4uL1N0YWNrL1N0YWNrXCI7XG5cbmNvbnN0IEJPUkRFUl9CT1RUT01fV0lEVEggPSA0O1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PHtcbiAgaG9yaXpvbnRhbFBhZGRpbmc6IFNwYWNlU2l6ZXMgfCBNUTxTcGFjZVNpemVzPjtcbn0+KCh7IHRoZW1lLCBob3Jpem9udGFsUGFkZGluZyB9KSA9PiAoe1xuICBkaXNwbGF5OiBcImZsZXhcIixcbiAgZmxleERpcmVjdGlvbjogXCJyb3dcIixcbiAgLi4ubXEoe1xuICAgIGdhcDogW1tcIm1cIiwgXCJsXCIsIFwibFwiXSwgdGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZ10sXG4gICAgcGFkZGluZ0lubGluZTogW2hvcml6b250YWxQYWRkaW5nLCB0aGVtZS52YXJpYWJsZXMuc2l6ZS5zcGFjaW5nXSxcbiAgfSksXG5cbiAgcGFkZGluZ1RvcDogdGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZy5tLFxuICBib3JkZXJCb3R0b206IGAxcHggc29saWQgJHt0aGVtZS52YWx1ZXMuY29sb3IuYm9yZGVyLnNlY29uZGFyeS5kZWZhdWx0fWAsXG5cbiAgd2hpdGVTcGFjZTogXCJub3dyYXBcIixcblxuICAvLyBTY3JvbGxiYXJcbiAgb3ZlcmZsb3c6IFwiYXV0b1wiLFxuICBzY3JvbGxiYXJXaWR0aDogXCJub25lXCIsIC8vIEZpcmVmb3hcbiAgbXNPdmVyZmxvd1N0eWxlOiBcIm5vbmVcIiwgLy8gSUUgMTArXG4gIFwiOjotd2Via2l0LXNjcm9sbGJhclwiOiB7XG4gICAgZGlzcGxheTogXCJub25lXCIsIC8vIENocm9tZVxuICB9LFxufSkpO1xuXG5jb25zdCBTdHlsZWRUYWIgPSBzdHlsZWQuYnV0dG9uPHsgYWN0aXZlOiBib29sZWFuIH0+
|
|
47
|
+
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlRhYnMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQVl3QiIsImZpbGUiOiJUYWJzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIHJlYWN0L2pzeC1wcm9wcy1uby1zcHJlYWRpbmcgKi9cblxuaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QgZnJvbSBcInJlYWN0XCI7XG5cbmltcG9ydCB7IG1xIH0gZnJvbSBcIi4uLy4uL3NoYXJlZC9tZWRpYVF1ZXJpZXNcIjtcbmltcG9ydCB7IE1RLCBTcGFjZVNpemVzIH0gZnJvbSBcIi4uLy4uL3R5cGVzXCI7XG5pbXBvcnQgeyBJY29uLCBJY29uTmFtZSB9IGZyb20gXCIuLi9JY29uL0ljb25cIjtcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSBcIi4uL1N0YWNrL1N0YWNrXCI7XG5cbmNvbnN0IEJPUkRFUl9CT1RUT01fV0lEVEggPSA0O1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PHtcbiAgaG9yaXpvbnRhbFBhZGRpbmc6IFNwYWNlU2l6ZXMgfCBNUTxTcGFjZVNpemVzPjtcbn0+KCh7IHRoZW1lLCBob3Jpem9udGFsUGFkZGluZyB9KSA9PiAoe1xuICBkaXNwbGF5OiBcImZsZXhcIixcbiAgZmxleERpcmVjdGlvbjogXCJyb3dcIixcbiAgLi4ubXEoe1xuICAgIGdhcDogW1tcIm1cIiwgXCJsXCIsIFwibFwiXSwgdGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZ10sXG4gICAgcGFkZGluZ0lubGluZTogW2hvcml6b250YWxQYWRkaW5nLCB0aGVtZS52YXJpYWJsZXMuc2l6ZS5zcGFjaW5nXSxcbiAgfSksXG5cbiAgcGFkZGluZ1RvcDogdGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZy5tLFxuICBib3JkZXJCb3R0b206IGAxcHggc29saWQgJHt0aGVtZS52YWx1ZXMuY29sb3IuYm9yZGVyLnNlY29uZGFyeS5kZWZhdWx0fWAsXG5cbiAgd2hpdGVTcGFjZTogXCJub3dyYXBcIixcblxuICAvLyBTY3JvbGxiYXJcbiAgb3ZlcmZsb3c6IFwiYXV0b1wiLFxuICBzY3JvbGxiYXJXaWR0aDogXCJub25lXCIsIC8vIEZpcmVmb3hcbiAgbXNPdmVyZmxvd1N0eWxlOiBcIm5vbmVcIiwgLy8gSUUgMTArXG4gIFwiOjotd2Via2l0LXNjcm9sbGJhclwiOiB7XG4gICAgZGlzcGxheTogXCJub25lXCIsIC8vIENocm9tZVxuICB9LFxufSkpO1xuXG5jb25zdCBTdHlsZWRUYWIgPSBzdHlsZWQuYnV0dG9uPHsgYWN0aXZlOiBib29sZWFuIH0+KCh7IHRoZW1lLCBhY3RpdmUgfSkgPT4gKHtcbiAgZGlzcGxheTogXCJmbGV4XCIsXG4gIGdhcDogdGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZy54eHMsXG4gIGFsaWduSXRlbXM6IFwiY2VudGVyXCIsXG5cbiAgYmFja2dyb3VuZDogMCxcbiAgYm9yZGVyOiAwLFxuICBib3hTaXppbmc6IFwiYm9yZGVyLWJveFwiLFxuICBjb2xvcjogdGhlbWUudmFsdWVzLmNvbG9yLnRleHQuc2Vjb25kYXJ5LmRlZmF1bHQsXG4gIHBhZGRpbmc6IDAsXG4gIGZvbnRGYW1pbHk6IHRoZW1lLnZhcmlhYmxlcy5mb250RmFtaWx5LmxhdG8sXG4gIGZvbnRXZWlnaHQ6IHRoZW1lLnZhcmlhYmxlcy53ZWlnaHQuYm9sZCxcbiAgZm9udFNpemU6IHRoZW1lLnZhcmlhYmxlcy5zaXplLmZvbnQudGV4dC5zLFxuICBsaW5lSGVpZ2h0OiB0aGVtZS52YXJpYWJsZXMuc2l6ZS5saW5lSGVpZ2h0LnRleHQucyxcbiAgdHJhbnNpdGlvbjogXCJiYWNrZ3JvdW5kLWNvbG9yIDAuMTVzXCIsXG4gIHRleHREZWNvcmF0aW9uOiBcIm5vbmVcIixcblxuICAvLyBSZXF1aXJlZCB0byBtYWludGFpbiBjb25zaXN0ZW50IGhlaWdodCBmb3IgdGhlIHNlbGVjdGVkIHRhYlxuICBib3JkZXJCb3R0b206IGAke0JPUkRFUl9CT1RUT01fV0lEVEh9cHggc29saWQgdHJhbnNwYXJlbnRgLFxuICBwYWRkaW5nQm90dG9tOiBgY2FsYygke3RoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmcubX0gLSAke0JPUkRFUl9CT1RUT01fV0lEVEh9cHgpYCxcblxuICAuLi4oYWN0aXZlICYmIHtcbiAgICBjb2xvcjogdGhlbWUudmFsdWVzLmNvbG9yLnRleHQuYWNjZW50LmRlZmF1bHQsXG4gICAgYm9yZGVyQm90dG9tOiBgJHtCT1JERVJfQk9UVE9NX1dJRFRIfXB4IHNvbGlkICR7dGhlbWUudmFsdWVzLmNvbG9yLmJhY2tncm91bmQuYWNjZW50LmRlZmF1bHR9YCxcbiAgfSksXG4gIC4uLighYWN0aXZlICYmIHtcbiAgICBjdXJzb3I6IFwicG9pbnRlclwiLFxuICAgIFwiJjpob3ZlclwiOiB7XG4gICAgICBjb2xvcjogdGhlbWUudmFsdWVzLmNvbG9yLnRleHQucHJpbWFyeS5kZWZhdWx0LFxuICAgICAgYm9yZGVyQm90dG9tOiBgJHtCT1JERVJfQk9UVE9NX1dJRFRIfXB4IHNvbGlkICR7dGhlbWUudmFsdWVzLmNvbG9yLmJhY2tncm91bmQudHJhbnNwYXJlbnQuaG92ZXJ9YCxcbiAgICB9LFxuICAgIFwiJjphY3RpdmVcIjoge1xuICAgICAgY29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci50ZXh0LnByaW1hcnkuZGVmYXVsdCxcbiAgICAgIGJvcmRlckJvdHRvbTogYCR7Qk9SREVSX0JPVFRPTV9XSURUSH1weCAgc29saWQgJHt0aGVtZS52YWx1ZXMuY29sb3IuYmFja2dyb3VuZC50cmFuc3BhcmVudC5zZWxlY3RlZH1gLFxuICAgIH0sXG4gIH0pLFxufSkpO1xuXG5jb25zdCBhZGRTdWJ0aXRsZSA9ICh0ZXh0OiBzdHJpbmcpID0+ICh0ZXh0ID8gYCgke3RleHR9KWAgOiBgYCk7XG5cbi8qKlxuICogQHBhcmFtIGxhYmVsIC0gVGhlIHRleHQgdG8gZGlzcGxheSBpbiB0aGUgdGFiXG4gKiBAcGFyYW0gc3VibGFiZWwgLSBBIHN1YnRleHQsIGVuY2FzZWQgaW4gcGFyZW50aGVzZXNcbiAqIEBwYXJhbSBpY29uIC0gQW4gaWNvbiB0byBkaXNwbGF5IHRvIHRoZSBsZWZ0IG9mIHRoZSBsYWJlbFxuICogQHBhcmFtIGFzIC0gVGhlIGVsZW1lbnQgdG8gcmVuZGVyIHRoZSB0YWIgYXMsIGRlZmF1bHRzIHRvIGBidXR0b25gXG4gKi9cbnR5cGUgVGFiID0ge1xuICBsYWJlbDogc3RyaW5nO1xuICBzdWJsYWJlbD86IHN0cmluZztcbiAgaWNvbj86IEljb25OYW1lO1xuICBhcz86IFwiYVwiIHwgXCJidXR0b25cIiB8IFJlYWN0LkNvbXBvbmVudFR5cGU7XG59ICYgUGFydGlhbDxSZWFjdC5Db21wb25lbnRQcm9wczxcImJ1dHRvblwiPj4gJlxuICBQYXJ0aWFsPFJlYWN0LkNvbXBvbmVudFByb3BzPFwiYVwiPj47XG5cbmV4cG9ydCB0eXBlIFRhYnNQcm9wcyA9IHtcbiAgY2hpbGRyZW4/OiBSZWFjdC5SZWFjdEVsZW1lbnQ7XG4gIC8qKiBUaGUgaW5kZXggb2YgdGhlIGFjdGl2ZSB0YWIgKi9cbiAgYWN0aXZlVGFiPzogbnVtYmVyO1xuICBvblRhYlNlbGVjdD86IChzZWxlY3RlZFRhYjogbnVtYmVyKSA9PiB2b2lkO1xuICAvKiogQW4gYXJyYXkgb2YgdGFicywgY29udGFpbnMgYGhlYWRlcmAsIGBzdWJ0aXRsZWAsIGBpY29uTGVmdGAsIGBhc2AgKGRlZmluaW5nIHRoZSB0YWIgYXMgYSBidXR0b24gb3IgYSBsaW5rKSAqL1xuICB0YWJzOiBUYWJbXTtcbiAgLyoqIEhvcml6b250YWwgcGFkZGluZyBzcGFjZSBmb3IgdGhlIHRhYiBjb250YWluZXIsIHN1aXRlZCBmb3IgdGFicyB3aXRoaW4gYSBDYXJkIGVsZW1lbnQgKi9cbiAgaG9yaXpvbnRhbFBhZGRpbmc/OiBTcGFjZVNpemVzIHwgTVE8U3BhY2VTaXplcz47IC8vICdtJyB8IFsnbScsICdsJywgJ2wnXVxuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBUYWJzKHtcbiAgdGFicyxcbiAgaG9yaXpvbnRhbFBhZGRpbmcsXG4gIGFjdGl2ZVRhYiA9IDAsXG4gIGNoaWxkcmVuLFxuICBvblRhYlNlbGVjdCxcbiAgXCJkYXRhLWUyZS10ZXN0LWlkXCI6IGRhdGFFMmVUZXN0SWQsXG59OiBUYWJzUHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICByZXR1cm4gKFxuICAgIDxTdGFjayBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfSBkYXRhLWRzLWlkPVwiVGFic1wiPlxuICAgICAgPFN0eWxlZENvbnRhaW5lciBob3Jpem9udGFsUGFkZGluZz17aG9yaXpvbnRhbFBhZGRpbmd9PlxuICAgICAgICB7dGFicy5tYXAoKHsgbGFiZWwsIHN1YmxhYmVsLCBpY29uLCBhcyA9IFwiYnV0dG9uXCIsIC4uLnJlc3QgfSwgaSkgPT4gKFxuICAgICAgICAgIDxTdHlsZWRUYWJcbiAgICAgICAgICAgIHR5cGU9e2FzICE9PSBcImJ1dHRvblwiID8gdW5kZWZpbmVkIDogXCJidXR0b25cIn1cbiAgICAgICAgICAgIGFzPXthc31cbiAgICAgICAgICAgIGtleT17bGFiZWx9XG4gICAgICAgICAgICBhY3RpdmU9e2FjdGl2ZVRhYiA9PT0gaX1cbiAgICAgICAgICAgIG9uQ2xpY2s9eygpID0+IG9uVGFiU2VsZWN0ICYmIG9uVGFiU2VsZWN0KGkpfVxuICAgICAgICAgICAgey4uLnJlc3R9XG4gICAgICAgICAgPlxuICAgICAgICAgICAge2ljb24gJiYgPEljb24gc2l6ZT1cInNcIiBuYW1lPXtpY29ufSAvPn1cbiAgICAgICAgICAgIHtsYWJlbH0ge2FkZFN1YnRpdGxlKHN1YmxhYmVsKX1cbiAgICAgICAgICA8L1N0eWxlZFRhYj5cbiAgICAgICAgKSl9XG4gICAgICA8L1N0eWxlZENvbnRhaW5lcj5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L1N0YWNrPlxuICApO1xufVxuIl19 */");
|
|
48
48
|
const StyledTab = /*#__PURE__*/_styled__default.default("button", process.env.NODE_ENV === "production" ? {
|
|
49
49
|
target: "epth00"
|
|
50
50
|
} : {
|
|
@@ -85,11 +85,11 @@ const StyledTab = /*#__PURE__*/_styled__default.default("button", process.env.NO
|
|
|
85
85
|
},
|
|
86
86
|
"&:active": {
|
|
87
87
|
color: theme.values.color.text.primary.default,
|
|
88
|
-
borderBottom: `${BORDER_BOTTOM_WIDTH}px solid ${theme.values.color.background.
|
|
88
|
+
borderBottom: `${BORDER_BOTTOM_WIDTH}px solid ${theme.values.color.background.transparent.selected}`
|
|
89
89
|
}
|
|
90
90
|
})
|
|
91
91
|
};
|
|
92
|
-
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlRhYnMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW9Da0IiLCJmaWxlIjoiVGFicy50c3giLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSByZWFjdC9qc3gtcHJvcHMtbm8tc3ByZWFkaW5nICovXG5cbmltcG9ydCBzdHlsZWQgZnJvbSBcIkBlbW90aW9uL3N0eWxlZFwiO1xuaW1wb3J0IFJlYWN0IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBtcSB9IGZyb20gXCIuLi8uLi9zaGFyZWQvbWVkaWFRdWVyaWVzXCI7XG5pbXBvcnQgeyBNUSwgU3BhY2VTaXplcyB9IGZyb20gXCIuLi8uLi90eXBlc1wiO1xuaW1wb3J0IHsgSWNvbiwgSWNvbk5hbWUgfSBmcm9tIFwiLi4vSWNvbi9JY29uXCI7XG5pbXBvcnQgeyBTdGFjayB9IGZyb20gXCIuLi9TdGFjay9TdGFja1wiO1xuXG5jb25zdCBCT1JERVJfQk9UVE9NX1dJRFRIID0gNDtcblxuY29uc3QgU3R5bGVkQ29udGFpbmVyID0gc3R5bGVkLmRpdjx7XG4gIGhvcml6b250YWxQYWRkaW5nOiBTcGFjZVNpemVzIHwgTVE8U3BhY2VTaXplcz47XG59PigoeyB0aGVtZSwgaG9yaXpvbnRhbFBhZGRpbmcgfSkgPT4gKHtcbiAgZGlzcGxheTogXCJmbGV4XCIsXG4gIGZsZXhEaXJlY3Rpb246IFwicm93XCIsXG4gIC4uLm1xKHtcbiAgICBnYXA6IFtbXCJtXCIsIFwibFwiLCBcImxcIl0sIHRoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmddLFxuICAgIHBhZGRpbmdJbmxpbmU6IFtob3Jpem9udGFsUGFkZGluZywgdGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZ10sXG4gIH0pLFxuXG4gIHBhZGRpbmdUb3A6IHRoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmcubSxcbiAgYm9yZGVyQm90dG9tOiBgMXB4IHNvbGlkICR7dGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5zZWNvbmRhcnkuZGVmYXVsdH1gLFxuXG4gIHdoaXRlU3BhY2U6IFwibm93cmFwXCIsXG5cbiAgLy8gU2Nyb2xsYmFyXG4gIG92ZXJmbG93OiBcImF1dG9cIixcbiAgc2Nyb2xsYmFyV2lkdGg6IFwibm9uZVwiLCAvLyBGaXJlZm94XG4gIG1zT3ZlcmZsb3dTdHlsZTogXCJub25lXCIsIC8vIElFIDEwK1xuICBcIjo6LXdlYmtpdC1zY3JvbGxiYXJcIjoge1xuICAgIGRpc3BsYXk6IFwibm9uZVwiLCAvLyBDaHJvbWVcbiAgfSxcbn0pKTtcblxuY29uc3QgU3R5bGVkVGFiID0gc3R5bGVkLmJ1dHRvbjx7IGFjdGl2ZTogYm9vbGVhbiB9PigoeyB0aGVtZSwgYWN0aXZlIH0pID0+
|
|
92
|
+
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlRhYnMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW9Da0IiLCJmaWxlIjoiVGFicy50c3giLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSByZWFjdC9qc3gtcHJvcHMtbm8tc3ByZWFkaW5nICovXG5cbmltcG9ydCBzdHlsZWQgZnJvbSBcIkBlbW90aW9uL3N0eWxlZFwiO1xuaW1wb3J0IFJlYWN0IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBtcSB9IGZyb20gXCIuLi8uLi9zaGFyZWQvbWVkaWFRdWVyaWVzXCI7XG5pbXBvcnQgeyBNUSwgU3BhY2VTaXplcyB9IGZyb20gXCIuLi8uLi90eXBlc1wiO1xuaW1wb3J0IHsgSWNvbiwgSWNvbk5hbWUgfSBmcm9tIFwiLi4vSWNvbi9JY29uXCI7XG5pbXBvcnQgeyBTdGFjayB9IGZyb20gXCIuLi9TdGFjay9TdGFja1wiO1xuXG5jb25zdCBCT1JERVJfQk9UVE9NX1dJRFRIID0gNDtcblxuY29uc3QgU3R5bGVkQ29udGFpbmVyID0gc3R5bGVkLmRpdjx7XG4gIGhvcml6b250YWxQYWRkaW5nOiBTcGFjZVNpemVzIHwgTVE8U3BhY2VTaXplcz47XG59PigoeyB0aGVtZSwgaG9yaXpvbnRhbFBhZGRpbmcgfSkgPT4gKHtcbiAgZGlzcGxheTogXCJmbGV4XCIsXG4gIGZsZXhEaXJlY3Rpb246IFwicm93XCIsXG4gIC4uLm1xKHtcbiAgICBnYXA6IFtbXCJtXCIsIFwibFwiLCBcImxcIl0sIHRoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmddLFxuICAgIHBhZGRpbmdJbmxpbmU6IFtob3Jpem9udGFsUGFkZGluZywgdGhlbWUudmFyaWFibGVzLnNpemUuc3BhY2luZ10sXG4gIH0pLFxuXG4gIHBhZGRpbmdUb3A6IHRoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmcubSxcbiAgYm9yZGVyQm90dG9tOiBgMXB4IHNvbGlkICR7dGhlbWUudmFsdWVzLmNvbG9yLmJvcmRlci5zZWNvbmRhcnkuZGVmYXVsdH1gLFxuXG4gIHdoaXRlU3BhY2U6IFwibm93cmFwXCIsXG5cbiAgLy8gU2Nyb2xsYmFyXG4gIG92ZXJmbG93OiBcImF1dG9cIixcbiAgc2Nyb2xsYmFyV2lkdGg6IFwibm9uZVwiLCAvLyBGaXJlZm94XG4gIG1zT3ZlcmZsb3dTdHlsZTogXCJub25lXCIsIC8vIElFIDEwK1xuICBcIjo6LXdlYmtpdC1zY3JvbGxiYXJcIjoge1xuICAgIGRpc3BsYXk6IFwibm9uZVwiLCAvLyBDaHJvbWVcbiAgfSxcbn0pKTtcblxuY29uc3QgU3R5bGVkVGFiID0gc3R5bGVkLmJ1dHRvbjx7IGFjdGl2ZTogYm9vbGVhbiB9PigoeyB0aGVtZSwgYWN0aXZlIH0pID0+ICh7XG4gIGRpc3BsYXk6IFwiZmxleFwiLFxuICBnYXA6IHRoZW1lLnZhcmlhYmxlcy5zaXplLnNwYWNpbmcueHhzLFxuICBhbGlnbkl0ZW1zOiBcImNlbnRlclwiLFxuXG4gIGJhY2tncm91bmQ6IDAsXG4gIGJvcmRlcjogMCxcbiAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgY29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci50ZXh0LnNlY29uZGFyeS5kZWZhdWx0LFxuICBwYWRkaW5nOiAwLFxuICBmb250RmFtaWx5OiB0aGVtZS52YXJpYWJsZXMuZm9udEZhbWlseS5sYXRvLFxuICBmb250V2VpZ2h0OiB0aGVtZS52YXJpYWJsZXMud2VpZ2h0LmJvbGQsXG4gIGZvbnRTaXplOiB0aGVtZS52YXJpYWJsZXMuc2l6ZS5mb250LnRleHQucyxcbiAgbGluZUhlaWdodDogdGhlbWUudmFyaWFibGVzLnNpemUubGluZUhlaWdodC50ZXh0LnMsXG4gIHRyYW5zaXRpb246IFwiYmFja2dyb3VuZC1jb2xvciAwLjE1c1wiLFxuICB0ZXh0RGVjb3JhdGlvbjogXCJub25lXCIsXG5cbiAgLy8gUmVxdWlyZWQgdG8gbWFpbnRhaW4gY29uc2lzdGVudCBoZWlnaHQgZm9yIHRoZSBzZWxlY3RlZCB0YWJcbiAgYm9yZGVyQm90dG9tOiBgJHtCT1JERVJfQk9UVE9NX1dJRFRIfXB4IHNvbGlkIHRyYW5zcGFyZW50YCxcbiAgcGFkZGluZ0JvdHRvbTogYGNhbGMoJHt0aGVtZS52YXJpYWJsZXMuc2l6ZS5zcGFjaW5nLm19IC0gJHtCT1JERVJfQk9UVE9NX1dJRFRIfXB4KWAsXG5cbiAgLi4uKGFjdGl2ZSAmJiB7XG4gICAgY29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci50ZXh0LmFjY2VudC5kZWZhdWx0LFxuICAgIGJvcmRlckJvdHRvbTogYCR7Qk9SREVSX0JPVFRPTV9XSURUSH1weCBzb2xpZCAke3RoZW1lLnZhbHVlcy5jb2xvci5iYWNrZ3JvdW5kLmFjY2VudC5kZWZhdWx0fWAsXG4gIH0pLFxuICAuLi4oIWFjdGl2ZSAmJiB7XG4gICAgY3Vyc29yOiBcInBvaW50ZXJcIixcbiAgICBcIiY6aG92ZXJcIjoge1xuICAgICAgY29sb3I6IHRoZW1lLnZhbHVlcy5jb2xvci50ZXh0LnByaW1hcnkuZGVmYXVsdCxcbiAgICAgIGJvcmRlckJvdHRvbTogYCR7Qk9SREVSX0JPVFRPTV9XSURUSH1weCBzb2xpZCAke3RoZW1lLnZhbHVlcy5jb2xvci5iYWNrZ3JvdW5kLnRyYW5zcGFyZW50LmhvdmVyfWAsXG4gICAgfSxcbiAgICBcIiY6YWN0aXZlXCI6IHtcbiAgICAgIGNvbG9yOiB0aGVtZS52YWx1ZXMuY29sb3IudGV4dC5wcmltYXJ5LmRlZmF1bHQsXG4gICAgICBib3JkZXJCb3R0b206IGAke0JPUkRFUl9CT1RUT01fV0lEVEh9cHggIHNvbGlkICR7dGhlbWUudmFsdWVzLmNvbG9yLmJhY2tncm91bmQudHJhbnNwYXJlbnQuc2VsZWN0ZWR9YCxcbiAgICB9LFxuICB9KSxcbn0pKTtcblxuY29uc3QgYWRkU3VidGl0bGUgPSAodGV4dDogc3RyaW5nKSA9PiAodGV4dCA/IGAoJHt0ZXh0fSlgIDogYGApO1xuXG4vKipcbiAqIEBwYXJhbSBsYWJlbCAtIFRoZSB0ZXh0IHRvIGRpc3BsYXkgaW4gdGhlIHRhYlxuICogQHBhcmFtIHN1YmxhYmVsIC0gQSBzdWJ0ZXh0LCBlbmNhc2VkIGluIHBhcmVudGhlc2VzXG4gKiBAcGFyYW0gaWNvbiAtIEFuIGljb24gdG8gZGlzcGxheSB0byB0aGUgbGVmdCBvZiB0aGUgbGFiZWxcbiAqIEBwYXJhbSBhcyAtIFRoZSBlbGVtZW50IHRvIHJlbmRlciB0aGUgdGFiIGFzLCBkZWZhdWx0cyB0byBgYnV0dG9uYFxuICovXG50eXBlIFRhYiA9IHtcbiAgbGFiZWw6IHN0cmluZztcbiAgc3VibGFiZWw/OiBzdHJpbmc7XG4gIGljb24/OiBJY29uTmFtZTtcbiAgYXM/OiBcImFcIiB8IFwiYnV0dG9uXCIgfCBSZWFjdC5Db21wb25lbnRUeXBlO1xufSAmIFBhcnRpYWw8UmVhY3QuQ29tcG9uZW50UHJvcHM8XCJidXR0b25cIj4+ICZcbiAgUGFydGlhbDxSZWFjdC5Db21wb25lbnRQcm9wczxcImFcIj4+O1xuXG5leHBvcnQgdHlwZSBUYWJzUHJvcHMgPSB7XG4gIGNoaWxkcmVuPzogUmVhY3QuUmVhY3RFbGVtZW50O1xuICAvKiogVGhlIGluZGV4IG9mIHRoZSBhY3RpdmUgdGFiICovXG4gIGFjdGl2ZVRhYj86IG51bWJlcjtcbiAgb25UYWJTZWxlY3Q/OiAoc2VsZWN0ZWRUYWI6IG51bWJlcikgPT4gdm9pZDtcbiAgLyoqIEFuIGFycmF5IG9mIHRhYnMsIGNvbnRhaW5zIGBoZWFkZXJgLCBgc3VidGl0bGVgLCBgaWNvbkxlZnRgLCBgYXNgIChkZWZpbmluZyB0aGUgdGFiIGFzIGEgYnV0dG9uIG9yIGEgbGluaykgKi9cbiAgdGFiczogVGFiW107XG4gIC8qKiBIb3Jpem9udGFsIHBhZGRpbmcgc3BhY2UgZm9yIHRoZSB0YWIgY29udGFpbmVyLCBzdWl0ZWQgZm9yIHRhYnMgd2l0aGluIGEgQ2FyZCBlbGVtZW50ICovXG4gIGhvcml6b250YWxQYWRkaW5nPzogU3BhY2VTaXplcyB8IE1RPFNwYWNlU2l6ZXM+OyAvLyAnbScgfCBbJ20nLCAnbCcsICdsJ11cbiAgXCJkYXRhLWUyZS10ZXN0LWlkXCI/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gVGFicyh7XG4gIHRhYnMsXG4gIGhvcml6b250YWxQYWRkaW5nLFxuICBhY3RpdmVUYWIgPSAwLFxuICBjaGlsZHJlbixcbiAgb25UYWJTZWxlY3QsXG4gIFwiZGF0YS1lMmUtdGVzdC1pZFwiOiBkYXRhRTJlVGVzdElkLFxufTogVGFic1Byb3BzKTogUmVhY3QuUmVhY3RFbGVtZW50IHtcbiAgcmV0dXJuIChcbiAgICA8U3RhY2sgZGF0YS1lMmUtdGVzdC1pZD17ZGF0YUUyZVRlc3RJZH0gZGF0YS1kcy1pZD1cIlRhYnNcIj5cbiAgICAgIDxTdHlsZWRDb250YWluZXIgaG9yaXpvbnRhbFBhZGRpbmc9e2hvcml6b250YWxQYWRkaW5nfT5cbiAgICAgICAge3RhYnMubWFwKCh7IGxhYmVsLCBzdWJsYWJlbCwgaWNvbiwgYXMgPSBcImJ1dHRvblwiLCAuLi5yZXN0IH0sIGkpID0+IChcbiAgICAgICAgICA8U3R5bGVkVGFiXG4gICAgICAgICAgICB0eXBlPXthcyAhPT0gXCJidXR0b25cIiA/IHVuZGVmaW5lZCA6IFwiYnV0dG9uXCJ9XG4gICAgICAgICAgICBhcz17YXN9XG4gICAgICAgICAgICBrZXk9e2xhYmVsfVxuICAgICAgICAgICAgYWN0aXZlPXthY3RpdmVUYWIgPT09IGl9XG4gICAgICAgICAgICBvbkNsaWNrPXsoKSA9PiBvblRhYlNlbGVjdCAmJiBvblRhYlNlbGVjdChpKX1cbiAgICAgICAgICAgIHsuLi5yZXN0fVxuICAgICAgICAgID5cbiAgICAgICAgICAgIHtpY29uICYmIDxJY29uIHNpemU9XCJzXCIgbmFtZT17aWNvbn0gLz59XG4gICAgICAgICAgICB7bGFiZWx9IHthZGRTdWJ0aXRsZShzdWJsYWJlbCl9XG4gICAgICAgICAgPC9TdHlsZWRUYWI+XG4gICAgICAgICkpfVxuICAgICAgPC9TdHlsZWRDb250YWluZXI+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9TdGFjaz5cbiAgKTtcbn1cbiJdfQ== */");
|
|
93
93
|
const addSubtitle = text => text ? `(${text})` : ``;
|
|
94
94
|
|
|
95
95
|
/**
|
|
@@ -1,113 +1,15 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
|
-
var _styled = require('@emotion/styled/base');
|
|
4
3
|
var React = require('react');
|
|
5
|
-
var react = require('@emotion/react');
|
|
6
|
-
var ReactDOM = require('react-dom');
|
|
7
|
-
var useDocument = require('../../shared/useDocument.js');
|
|
8
|
-
var useWindow = require('../../shared/useWindow.js');
|
|
9
|
-
var SubThemeProvider = require('../SubThemeProvider/SubThemeProvider.js');
|
|
10
4
|
var Text = require('../Typography/Text/Text.js');
|
|
11
|
-
var
|
|
12
|
-
var utils = require('./utils.js');
|
|
5
|
+
var TooltipContent = require('./TooltipContent.js');
|
|
13
6
|
|
|
14
7
|
function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
|
|
15
8
|
|
|
16
|
-
var _styled__default = /*#__PURE__*/_interopDefault(_styled);
|
|
17
9
|
var React__default = /*#__PURE__*/_interopDefault(React);
|
|
18
10
|
|
|
19
|
-
const ANIMATION_DURATION = 200;
|
|
20
11
|
const SHOW_HIDE_DELAY = 200;
|
|
21
|
-
|
|
22
|
-
target: "e2kei841"
|
|
23
|
-
} : {
|
|
24
|
-
target: "e2kei841",
|
|
25
|
-
label: "StyledContainer"
|
|
26
|
-
})(_ref => {
|
|
27
|
-
let {
|
|
28
|
-
theme,
|
|
29
|
-
horizontalPlacement,
|
|
30
|
-
verticalPlacement
|
|
31
|
-
} = _ref;
|
|
32
|
-
const animationDistance = verticalPlacement === "top" ? `${utils.ANIMATION_DISTANCE}px` : `-${utils.ANIMATION_DISTANCE}px`;
|
|
33
|
-
const animation = react.keyframes({
|
|
34
|
-
to: {
|
|
35
|
-
opacity: 1,
|
|
36
|
-
transform: horizontalPlacement === "center" ? `translate(-50%, ${animationDistance})` : `translateY(${animationDistance})`
|
|
37
|
-
}
|
|
38
|
-
});
|
|
39
|
-
return {
|
|
40
|
-
position: "absolute",
|
|
41
|
-
zIndex: _zindex.default.tooltip.value,
|
|
42
|
-
opacity: 0,
|
|
43
|
-
animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,
|
|
44
|
-
borderRadius: theme.variables.size.borderRadius.xs,
|
|
45
|
-
backgroundColor: theme.values.color.background.primary.default,
|
|
46
|
-
maxWidth: "224px",
|
|
47
|
-
boxSizing: "border-box",
|
|
48
|
-
padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,
|
|
49
|
-
...(horizontalPlacement === "center" && {
|
|
50
|
-
transform: "translate(-50%)"
|
|
51
|
-
})
|
|
52
|
-
};
|
|
53
|
-
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.tsx"],"names":[],"mappings":"AAyEwB","file":"Tooltip.tsx","sourcesContent":["import React, {\n  useState,\n  useRef,\n  useLayoutEffect,\n  useEffect,\n  useCallback,\n  MutableRefObject,\n  useMemo,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport { Text } from \"../Typography/Text/Text\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ARROW_SIZE,\n  ARROW_OFFSET,\n  ANIMATION_DISTANCE,\n  getTooltipPosition,\n} from \"./utils\";\n\nexport type BaseProps = {\n  /* Tooltip content */\n  content: string;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  \"data-e2e-test-id\"?: string;\n  /* Called when tooltip appears and disappears */\n  onVisibilityChange?: (isVisible: boolean) => void;\n};\n\nexport type ConditionalProps =\n  | {\n      externalTriggerRef?: never;\n      /* Trigger element - wrap trigger element within Tooltip - takes precedence over trigger prop */\n      children: React.ReactElement;\n    }\n  | {\n      children?: never;\n      /* Trigger element ref - pass in an external trigger element */\n      externalTriggerRef: MutableRefObject<any>;\n    };\n\nexport type TooltipProps = BaseProps & ConditionalProps;\n\nexport type TooltipPosition = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = {\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({ theme, horizontalPlacement, verticalPlacement }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      borderRadius: theme.variables.size.borderRadius.xs,\n      backgroundColor: theme.values.color.background.primary.default,\n      maxWidth: \"224px\",\n      boxSizing: \"border-box\",\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n};\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({ theme, verticalPlacement, horizontalPlacement }) => ({\n    position: \"absolute\",\n    width: 0,\n    height: 0,\n    borderLeft: `${ARROW_SIZE}px solid transparent`,\n    borderRight: `${ARROW_SIZE}px solid transparent`,\n\n    ...(verticalPlacement === \"top\" && {\n      top: \"100%\",\n      borderTop: `${ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`,\n    }),\n\n    ...(verticalPlacement === \"bottom\" && {\n      top: `-${ARROW_SIZE}px`,\n      borderBottom: `${ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`,\n    }),\n\n    ...(horizontalPlacement === \"center\" && {\n      left: \"50%\",\n      transform: \"translate(-50%)\",\n    }),\n\n    ...(horizontalPlacement === \"right\" && {\n      left: `${ARROW_OFFSET}px`,\n    }),\n\n    ...(horizontalPlacement === \"left\" && {\n      right: `${ARROW_OFFSET}px`,\n    }),\n  })\n);\n\nconst initialPosition: TooltipPosition = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\nexport function Tooltip({\n  placement = \"auto\",\n  content,\n  children,\n  externalTriggerRef,\n  portalContainer,\n  \"data-e2e-test-id\": dataE2eTestId,\n  onVisibilityChange,\n}: TooltipProps): React.ReactElement {\n  const tooltipId = useMemo(\n    () => `DSTooltip_${Math.floor(Date.now() * Math.random())}`,\n    []\n  );\n  const [position, setPosition] = useState(initialPosition);\n  const [isVisible, setVisible] = useState(false);\n  const triggeredByEvent = useRef(null); // indicates if triggered by hover or focus\n  const isTooltipHovered = useRef(false);\n  const isTriggerHovered = useRef(false);\n  const internalTriggerRef = useRef(null);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n  const hideTooltipTimeoutId = useRef(null);\n  const showTooltipTimeoutId = useRef(null);\n  const triggerRef = externalTriggerRef || internalTriggerRef;\n\n  const toggleVisibility = useCallback(\n    (status: boolean) => {\n      setVisible(status);\n\n      if (onVisibilityChange) {\n        onVisibilityChange(status);\n      }\n\n      // log time when tooltip closes\n      if (!status) {\n        lastTooltipHideTimestamp = Date.now();\n      }\n    },\n    [onVisibilityChange]\n  );\n\n  const handleTriggerPointerEnter = useCallback(() => {\n    isTriggerHovered.current = true;\n    if (!isTooltipHovered.current) {\n      clearTimeout(showTooltipTimeoutId.current);\n      // Delay show tooltip to prevent flickering when mouse moves quickly over trigger\n      showTooltipTimeoutId.current = setTimeout(() => {\n        if (isTriggerHovered.current) {\n          triggeredByEvent.current = \"hover\";\n          toggleVisibility(true);\n        }\n      }, SHOW_HIDE_DELAY);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerPointerLeave = useCallback(() => {\n    isTriggerHovered.current = false;\n    clearTimeout(hideTooltipTimeoutId.current);\n    // Delay removing tooltip from DOM to allow hover over tooltip element\n    hideTooltipTimeoutId.current = setTimeout(() => {\n      if (\n        !isTooltipHovered.current &&\n        triggeredByEvent.current === \"hover\" &&\n        !isTriggerHovered.current\n      ) {\n        toggleVisibility(false);\n      }\n    }, SHOW_HIDE_DELAY);\n  }, [toggleVisibility]);\n\n  const handleTriggerFocus = useCallback(() => {\n    triggeredByEvent.current = \"focus\";\n    toggleVisibility(true);\n  }, [toggleVisibility]);\n\n  const handleTriggerBlur = useCallback(() => {\n    if (triggeredByEvent.current === \"focus\") {\n      toggleVisibility(false);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerKeyDown = useCallback(\n    (evt) => {\n      if (evt.key === \"Escape\") {\n        toggleVisibility(false);\n      }\n    },\n    [toggleVisibility]\n  );\n\n  const handleTooltipPointerEnter = () => {\n    isTooltipHovered.current = true;\n  };\n\n  const handleTooltipPointerLeave = () => {\n    isTooltipHovered.current = false;\n    if (triggeredByEvent.current === \"hover\") {\n      toggleVisibility(false);\n    }\n  };\n\n  useEffect(\n    () => () => {\n      // clear timers\n      clearTimeout(showTooltipTimeoutId.current);\n      clearTimeout(hideTooltipTimeoutId.current);\n    },\n    []\n  );\n\n  useEffect(() => {\n    let trigger: HTMLElement;\n\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      trigger = externalTriggerRef.current;\n\n      trigger.setAttribute(\"tabindex\", \"0\");\n      trigger.addEventListener(\"pointerenter\", handleTriggerPointerEnter);\n      trigger.addEventListener(\"pointerleave\", handleTriggerPointerLeave);\n      trigger.addEventListener(\"focus\", handleTriggerFocus);\n      trigger.addEventListener(\"blur\", handleTriggerBlur);\n      trigger.addEventListener(\"keydown\", handleTriggerKeyDown);\n    }\n\n    return () => {\n      if (trigger) {\n        trigger.removeEventListener(\"pointerenter\", handleTriggerPointerEnter);\n        trigger.removeEventListener(\"pointerleave\", handleTriggerPointerLeave);\n        trigger.removeEventListener(\"focus\", handleTriggerFocus);\n        trigger.removeEventListener(\"blur\", handleTriggerBlur);\n        trigger.removeEventListener(\"keydown\", handleTriggerKeyDown);\n      }\n    };\n  }, [\n    externalTriggerRef,\n    children,\n    handleTriggerPointerEnter,\n    handleTriggerPointerLeave,\n    handleTriggerFocus,\n    handleTriggerBlur,\n    handleTriggerKeyDown,\n  ]);\n\n  useEffect(() => {\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      const trigger = externalTriggerRef.current;\n\n      if (isVisible) {\n        trigger.setAttribute(\"aria-describedby\", tooltipId);\n      } else {\n        trigger.removeAttribute(\"aria-describedby\");\n      }\n    }\n  }, [externalTriggerRef, children, tooltipId, isVisible]);\n\n  useLayoutEffect(() => {\n    if (isVisible && triggerRef.current && tooltipRef.current) {\n      // calculate tooltip position\n      setPosition(\n        getTooltipPosition(placement, triggerRef, tooltipRef, document, window)\n      );\n    }\n  }, [isVisible, placement, triggerRef, document, window]);\n\n  let portal;\n\n  if (isVisible) {\n    const tooltipElm = (\n      <SubThemeProvider name=\"inverted\">\n        <StyledContainer\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"Tooltip\"\n          style={{\n            top: position.top,\n            left: position.left,\n            animationDuration: getAnimationDuration(),\n          }}\n          ref={tooltipRef}\n          id={tooltipId}\n          role=\"tooltip\"\n          aria-hidden=\"true\"\n          horizontalPlacement={position.horizontalPlacement}\n          verticalPlacement={position.verticalPlacement}\n          onPointerEnter={handleTooltipPointerEnter}\n          onPointerLeave={handleTooltipPointerLeave}\n        >\n          <Text size=\"s\">{content}</Text>\n          <StyledArrow\n            horizontalPlacement={position.horizontalPlacement}\n            verticalPlacement={position.verticalPlacement}\n          />\n        </StyledContainer>\n      </SubThemeProvider>\n    );\n\n    portal = createPortal(tooltipElm, portalContainer || document.body);\n  }\n\n  const triggerElm = children\n    ? React.cloneElement(children, {\n        ref: triggerRef,\n        ...(isVisible && {\n          \"aria-describedby\": tooltipId,\n        }),\n        tabIndex: 0,\n        onPointerEnter: handleTriggerPointerEnter,\n        onPointerLeave: handleTriggerPointerLeave,\n        onFocus: handleTriggerFocus,\n        onBlur: handleTriggerBlur,\n        onKeyDown: handleTriggerKeyDown,\n      })\n    : null;\n\n  return (\n    <>\n      {triggerElm}\n      {portal}\n    </>\n  );\n}\n"]} */");
|
|
54
|
-
const StyledArrow = /*#__PURE__*/_styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
55
|
-
target: "e2kei840"
|
|
56
|
-
} : {
|
|
57
|
-
target: "e2kei840",
|
|
58
|
-
label: "StyledArrow"
|
|
59
|
-
})(_ref2 => {
|
|
60
|
-
let {
|
|
61
|
-
theme,
|
|
62
|
-
verticalPlacement,
|
|
63
|
-
horizontalPlacement
|
|
64
|
-
} = _ref2;
|
|
65
|
-
return {
|
|
66
|
-
position: "absolute",
|
|
67
|
-
width: 0,
|
|
68
|
-
height: 0,
|
|
69
|
-
borderLeft: `${utils.ARROW_SIZE}px solid transparent`,
|
|
70
|
-
borderRight: `${utils.ARROW_SIZE}px solid transparent`,
|
|
71
|
-
...(verticalPlacement === "top" && {
|
|
72
|
-
top: "100%",
|
|
73
|
-
borderTop: `${utils.ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`
|
|
74
|
-
}),
|
|
75
|
-
...(verticalPlacement === "bottom" && {
|
|
76
|
-
top: `-${utils.ARROW_SIZE}px`,
|
|
77
|
-
borderBottom: `${utils.ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`
|
|
78
|
-
}),
|
|
79
|
-
...(horizontalPlacement === "center" && {
|
|
80
|
-
left: "50%",
|
|
81
|
-
transform: "translate(-50%)"
|
|
82
|
-
}),
|
|
83
|
-
...(horizontalPlacement === "right" && {
|
|
84
|
-
left: `${utils.ARROW_OFFSET}px`
|
|
85
|
-
}),
|
|
86
|
-
...(horizontalPlacement === "left" && {
|
|
87
|
-
right: `${utils.ARROW_OFFSET}px`
|
|
88
|
-
})
|
|
89
|
-
};
|
|
90
|
-
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.tsx"],"names":[],"mappings":"AAgHoB","file":"Tooltip.tsx","sourcesContent":["import React, {\n  useState,\n  useRef,\n  useLayoutEffect,\n  useEffect,\n  useCallback,\n  MutableRefObject,\n  useMemo,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\nimport { createPortal } from \"react-dom\";\nimport { useDocument } from \"../../shared/useDocument\";\nimport { useWindow } from \"../../shared/useWindow\";\nimport { SubThemeProvider } from \"../SubThemeProvider/SubThemeProvider\";\nimport { Text } from \"../Typography/Text/Text\";\nimport zIndices from \"../../../build-tokens/_zindex.json\";\n\nimport {\n  ARROW_SIZE,\n  ARROW_OFFSET,\n  ANIMATION_DISTANCE,\n  getTooltipPosition,\n} from \"./utils\";\n\nexport type BaseProps = {\n  /* Tooltip content */\n  content: string;\n  /* Placement */\n  placement?:\n    | \"auto\"\n    | \"top\"\n    | \"bottom\"\n    | \"top-left\"\n    | \"top-right\"\n    | \"bottom-left\"\n    | \"bottom-right\";\n  /* Custom portal container to render tooltip into */\n  portalContainer?: HTMLElement;\n  \"data-e2e-test-id\"?: string;\n  /* Called when tooltip appears and disappears */\n  onVisibilityChange?: (isVisible: boolean) => void;\n};\n\nexport type ConditionalProps =\n  | {\n      externalTriggerRef?: never;\n      /* Trigger element - wrap trigger element within Tooltip - takes precedence over trigger prop */\n      children: React.ReactElement;\n    }\n  | {\n      children?: never;\n      /* Trigger element ref - pass in an external trigger element */\n      externalTriggerRef: MutableRefObject<any>;\n    };\n\nexport type TooltipProps = BaseProps & ConditionalProps;\n\nexport type TooltipPosition = {\n  top: number;\n  left: number;\n  horizontalPlacement: \"left\" | \"right\" | \"center\";\n  verticalPlacement: \"top\" | \"bottom\";\n};\n\ntype StyledContainerProps = {\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({ theme, horizontalPlacement, verticalPlacement }) => {\n    const animationDistance =\n      verticalPlacement === \"top\"\n        ? `${ANIMATION_DISTANCE}px`\n        : `-${ANIMATION_DISTANCE}px`;\n    const animation = keyframes({\n      to: {\n        opacity: 1,\n        transform:\n          horizontalPlacement === \"center\"\n            ? `translate(-50%, ${animationDistance})`\n            : `translateY(${animationDistance})`,\n      },\n    });\n\n    return {\n      position: \"absolute\",\n      zIndex: zIndices.tooltip.value,\n      opacity: 0,\n      animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,\n      borderRadius: theme.variables.size.borderRadius.xs,\n      backgroundColor: theme.values.color.background.primary.default,\n      maxWidth: \"224px\",\n      boxSizing: \"border-box\",\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipPosition[\"verticalPlacement\"];\n  horizontalPlacement: TooltipPosition[\"horizontalPlacement\"];\n};\n\nconst StyledArrow = styled.div<StyledArrowProps>(\n  ({ theme, verticalPlacement, horizontalPlacement }) => ({\n    position: \"absolute\",\n    width: 0,\n    height: 0,\n    borderLeft: `${ARROW_SIZE}px solid transparent`,\n    borderRight: `${ARROW_SIZE}px solid transparent`,\n\n    ...(verticalPlacement === \"top\" && {\n      top: \"100%\",\n      borderTop: `${ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`,\n    }),\n\n    ...(verticalPlacement === \"bottom\" && {\n      top: `-${ARROW_SIZE}px`,\n      borderBottom: `${ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`,\n    }),\n\n    ...(horizontalPlacement === \"center\" && {\n      left: \"50%\",\n      transform: \"translate(-50%)\",\n    }),\n\n    ...(horizontalPlacement === \"right\" && {\n      left: `${ARROW_OFFSET}px`,\n    }),\n\n    ...(horizontalPlacement === \"left\" && {\n      right: `${ARROW_OFFSET}px`,\n    }),\n  })\n);\n\nconst initialPosition: TooltipPosition = {\n  top: 0,\n  left: 0,\n  verticalPlacement: \"top\",\n  horizontalPlacement: \"center\",\n};\n\nlet lastTooltipHideTimestamp = 0;\n\n/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */\nfunction getAnimationDuration() {\n  let animationDuration = `${ANIMATION_DURATION}ms`;\n\n  if (lastTooltipHideTimestamp) {\n    const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;\n\n    if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {\n      animationDuration = \"0ms\";\n    }\n  }\n  return animationDuration;\n}\n\nexport function Tooltip({\n  placement = \"auto\",\n  content,\n  children,\n  externalTriggerRef,\n  portalContainer,\n  \"data-e2e-test-id\": dataE2eTestId,\n  onVisibilityChange,\n}: TooltipProps): React.ReactElement {\n  const tooltipId = useMemo(\n    () => `DSTooltip_${Math.floor(Date.now() * Math.random())}`,\n    []\n  );\n  const [position, setPosition] = useState(initialPosition);\n  const [isVisible, setVisible] = useState(false);\n  const triggeredByEvent = useRef(null); // indicates if triggered by hover or focus\n  const isTooltipHovered = useRef(false);\n  const isTriggerHovered = useRef(false);\n  const internalTriggerRef = useRef(null);\n  const tooltipRef = useRef(null);\n  const document = useDocument();\n  const window = useWindow();\n  const hideTooltipTimeoutId = useRef(null);\n  const showTooltipTimeoutId = useRef(null);\n  const triggerRef = externalTriggerRef || internalTriggerRef;\n\n  const toggleVisibility = useCallback(\n    (status: boolean) => {\n      setVisible(status);\n\n      if (onVisibilityChange) {\n        onVisibilityChange(status);\n      }\n\n      // log time when tooltip closes\n      if (!status) {\n        lastTooltipHideTimestamp = Date.now();\n      }\n    },\n    [onVisibilityChange]\n  );\n\n  const handleTriggerPointerEnter = useCallback(() => {\n    isTriggerHovered.current = true;\n    if (!isTooltipHovered.current) {\n      clearTimeout(showTooltipTimeoutId.current);\n      // Delay show tooltip to prevent flickering when mouse moves quickly over trigger\n      showTooltipTimeoutId.current = setTimeout(() => {\n        if (isTriggerHovered.current) {\n          triggeredByEvent.current = \"hover\";\n          toggleVisibility(true);\n        }\n      }, SHOW_HIDE_DELAY);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerPointerLeave = useCallback(() => {\n    isTriggerHovered.current = false;\n    clearTimeout(hideTooltipTimeoutId.current);\n    // Delay removing tooltip from DOM to allow hover over tooltip element\n    hideTooltipTimeoutId.current = setTimeout(() => {\n      if (\n        !isTooltipHovered.current &&\n        triggeredByEvent.current === \"hover\" &&\n        !isTriggerHovered.current\n      ) {\n        toggleVisibility(false);\n      }\n    }, SHOW_HIDE_DELAY);\n  }, [toggleVisibility]);\n\n  const handleTriggerFocus = useCallback(() => {\n    triggeredByEvent.current = \"focus\";\n    toggleVisibility(true);\n  }, [toggleVisibility]);\n\n  const handleTriggerBlur = useCallback(() => {\n    if (triggeredByEvent.current === \"focus\") {\n      toggleVisibility(false);\n    }\n  }, [toggleVisibility]);\n\n  const handleTriggerKeyDown = useCallback(\n    (evt) => {\n      if (evt.key === \"Escape\") {\n        toggleVisibility(false);\n      }\n    },\n    [toggleVisibility]\n  );\n\n  const handleTooltipPointerEnter = () => {\n    isTooltipHovered.current = true;\n  };\n\n  const handleTooltipPointerLeave = () => {\n    isTooltipHovered.current = false;\n    if (triggeredByEvent.current === \"hover\") {\n      toggleVisibility(false);\n    }\n  };\n\n  useEffect(\n    () => () => {\n      // clear timers\n      clearTimeout(showTooltipTimeoutId.current);\n      clearTimeout(hideTooltipTimeoutId.current);\n    },\n    []\n  );\n\n  useEffect(() => {\n    let trigger: HTMLElement;\n\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      trigger = externalTriggerRef.current;\n\n      trigger.setAttribute(\"tabindex\", \"0\");\n      trigger.addEventListener(\"pointerenter\", handleTriggerPointerEnter);\n      trigger.addEventListener(\"pointerleave\", handleTriggerPointerLeave);\n      trigger.addEventListener(\"focus\", handleTriggerFocus);\n      trigger.addEventListener(\"blur\", handleTriggerBlur);\n      trigger.addEventListener(\"keydown\", handleTriggerKeyDown);\n    }\n\n    return () => {\n      if (trigger) {\n        trigger.removeEventListener(\"pointerenter\", handleTriggerPointerEnter);\n        trigger.removeEventListener(\"pointerleave\", handleTriggerPointerLeave);\n        trigger.removeEventListener(\"focus\", handleTriggerFocus);\n        trigger.removeEventListener(\"blur\", handleTriggerBlur);\n        trigger.removeEventListener(\"keydown\", handleTriggerKeyDown);\n      }\n    };\n  }, [\n    externalTriggerRef,\n    children,\n    handleTriggerPointerEnter,\n    handleTriggerPointerLeave,\n    handleTriggerFocus,\n    handleTriggerBlur,\n    handleTriggerKeyDown,\n  ]);\n\n  useEffect(() => {\n    if (externalTriggerRef && externalTriggerRef.current && !children) {\n      const trigger = externalTriggerRef.current;\n\n      if (isVisible) {\n        trigger.setAttribute(\"aria-describedby\", tooltipId);\n      } else {\n        trigger.removeAttribute(\"aria-describedby\");\n      }\n    }\n  }, [externalTriggerRef, children, tooltipId, isVisible]);\n\n  useLayoutEffect(() => {\n    if (isVisible && triggerRef.current && tooltipRef.current) {\n      // calculate tooltip position\n      setPosition(\n        getTooltipPosition(placement, triggerRef, tooltipRef, document, window)\n      );\n    }\n  }, [isVisible, placement, triggerRef, document, window]);\n\n  let portal;\n\n  if (isVisible) {\n    const tooltipElm = (\n      <SubThemeProvider name=\"inverted\">\n        <StyledContainer\n          data-e2e-test-id={dataE2eTestId}\n          data-ds-id=\"Tooltip\"\n          style={{\n            top: position.top,\n            left: position.left,\n            animationDuration: getAnimationDuration(),\n          }}\n          ref={tooltipRef}\n          id={tooltipId}\n          role=\"tooltip\"\n          aria-hidden=\"true\"\n          horizontalPlacement={position.horizontalPlacement}\n          verticalPlacement={position.verticalPlacement}\n          onPointerEnter={handleTooltipPointerEnter}\n          onPointerLeave={handleTooltipPointerLeave}\n        >\n          <Text size=\"s\">{content}</Text>\n          <StyledArrow\n            horizontalPlacement={position.horizontalPlacement}\n            verticalPlacement={position.verticalPlacement}\n          />\n        </StyledContainer>\n      </SubThemeProvider>\n    );\n\n    portal = createPortal(tooltipElm, portalContainer || document.body);\n  }\n\n  const triggerElm = children\n    ? React.cloneElement(children, {\n        ref: triggerRef,\n        ...(isVisible && {\n          \"aria-describedby\": tooltipId,\n        }),\n        tabIndex: 0,\n        onPointerEnter: handleTriggerPointerEnter,\n        onPointerLeave: handleTriggerPointerLeave,\n        onFocus: handleTriggerFocus,\n        onBlur: handleTriggerBlur,\n        onKeyDown: handleTriggerKeyDown,\n      })\n    : null;\n\n  return (\n    <>\n      {triggerElm}\n      {portal}\n    </>\n  );\n}\n"]} */");
|
|
91
|
-
const initialPosition = {
|
|
92
|
-
top: 0,
|
|
93
|
-
left: 0,
|
|
94
|
-
verticalPlacement: "top",
|
|
95
|
-
horizontalPlacement: "center"
|
|
96
|
-
};
|
|
97
|
-
let lastTooltipHideTimestamp = 0;
|
|
98
|
-
|
|
99
|
-
/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */
|
|
100
|
-
function getAnimationDuration() {
|
|
101
|
-
let animationDuration = `${ANIMATION_DURATION}ms`;
|
|
102
|
-
if (lastTooltipHideTimestamp) {
|
|
103
|
-
const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;
|
|
104
|
-
if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {
|
|
105
|
-
animationDuration = "0ms";
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
return animationDuration;
|
|
109
|
-
}
|
|
110
|
-
function Tooltip(_ref3) {
|
|
12
|
+
function Tooltip(_ref) {
|
|
111
13
|
let {
|
|
112
14
|
placement = "auto",
|
|
113
15
|
content,
|
|
@@ -116,30 +18,21 @@ function Tooltip(_ref3) {
|
|
|
116
18
|
portalContainer,
|
|
117
19
|
"data-e2e-test-id": dataE2eTestId,
|
|
118
20
|
onVisibilityChange
|
|
119
|
-
} =
|
|
21
|
+
} = _ref;
|
|
120
22
|
const tooltipId = React.useMemo(() => `DSTooltip_${Math.floor(Date.now() * Math.random())}`, []);
|
|
121
|
-
const [position, setPosition] = React.useState(initialPosition);
|
|
122
23
|
const [isVisible, setVisible] = React.useState(false);
|
|
123
24
|
const triggeredByEvent = React.useRef(null); // indicates if triggered by hover or focus
|
|
124
25
|
const isTooltipHovered = React.useRef(false);
|
|
125
26
|
const isTriggerHovered = React.useRef(false);
|
|
126
|
-
const internalTriggerRef = React.useRef(null);
|
|
127
|
-
const tooltipRef = React.useRef(null);
|
|
128
|
-
const document = useDocument.useDocument();
|
|
129
|
-
const window = useWindow.useWindow();
|
|
130
27
|
const hideTooltipTimeoutId = React.useRef(null);
|
|
131
28
|
const showTooltipTimeoutId = React.useRef(null);
|
|
29
|
+
const internalTriggerRef = React.useRef(null);
|
|
132
30
|
const triggerRef = externalTriggerRef || internalTriggerRef;
|
|
133
31
|
const toggleVisibility = React.useCallback(status => {
|
|
134
32
|
setVisible(status);
|
|
135
33
|
if (onVisibilityChange) {
|
|
136
34
|
onVisibilityChange(status);
|
|
137
35
|
}
|
|
138
|
-
|
|
139
|
-
// log time when tooltip closes
|
|
140
|
-
if (!status) {
|
|
141
|
-
lastTooltipHideTimestamp = Date.now();
|
|
142
|
-
}
|
|
143
36
|
}, [onVisibilityChange]);
|
|
144
37
|
const handleTriggerPointerEnter = React.useCallback(() => {
|
|
145
38
|
isTriggerHovered.current = true;
|
|
@@ -194,7 +87,7 @@ function Tooltip(_ref3) {
|
|
|
194
87
|
}, []);
|
|
195
88
|
React.useEffect(() => {
|
|
196
89
|
let trigger;
|
|
197
|
-
if (externalTriggerRef
|
|
90
|
+
if (externalTriggerRef?.current && !children) {
|
|
198
91
|
trigger = externalTriggerRef.current;
|
|
199
92
|
trigger.setAttribute("tabindex", "0");
|
|
200
93
|
trigger.addEventListener("pointerenter", handleTriggerPointerEnter);
|
|
@@ -214,7 +107,7 @@ function Tooltip(_ref3) {
|
|
|
214
107
|
};
|
|
215
108
|
}, [externalTriggerRef, children, handleTriggerPointerEnter, handleTriggerPointerLeave, handleTriggerFocus, handleTriggerBlur, handleTriggerKeyDown]);
|
|
216
109
|
React.useEffect(() => {
|
|
217
|
-
if (externalTriggerRef
|
|
110
|
+
if (externalTriggerRef?.current && !children) {
|
|
218
111
|
const trigger = externalTriggerRef.current;
|
|
219
112
|
if (isVisible) {
|
|
220
113
|
trigger.setAttribute("aria-describedby", tooltipId);
|
|
@@ -223,40 +116,6 @@ function Tooltip(_ref3) {
|
|
|
223
116
|
}
|
|
224
117
|
}
|
|
225
118
|
}, [externalTriggerRef, children, tooltipId, isVisible]);
|
|
226
|
-
React.useLayoutEffect(() => {
|
|
227
|
-
if (isVisible && triggerRef.current && tooltipRef.current) {
|
|
228
|
-
// calculate tooltip position
|
|
229
|
-
setPosition(utils.getTooltipPosition(placement, triggerRef, tooltipRef, document, window));
|
|
230
|
-
}
|
|
231
|
-
}, [isVisible, placement, triggerRef, document, window]);
|
|
232
|
-
let portal;
|
|
233
|
-
if (isVisible) {
|
|
234
|
-
const tooltipElm = /*#__PURE__*/React__default.default.createElement(SubThemeProvider.SubThemeProvider, {
|
|
235
|
-
name: "inverted"
|
|
236
|
-
}, /*#__PURE__*/React__default.default.createElement(StyledContainer, {
|
|
237
|
-
"data-e2e-test-id": dataE2eTestId,
|
|
238
|
-
"data-ds-id": "Tooltip",
|
|
239
|
-
style: {
|
|
240
|
-
top: position.top,
|
|
241
|
-
left: position.left,
|
|
242
|
-
animationDuration: getAnimationDuration()
|
|
243
|
-
},
|
|
244
|
-
ref: tooltipRef,
|
|
245
|
-
id: tooltipId,
|
|
246
|
-
role: "tooltip",
|
|
247
|
-
"aria-hidden": "true",
|
|
248
|
-
horizontalPlacement: position.horizontalPlacement,
|
|
249
|
-
verticalPlacement: position.verticalPlacement,
|
|
250
|
-
onPointerEnter: handleTooltipPointerEnter,
|
|
251
|
-
onPointerLeave: handleTooltipPointerLeave
|
|
252
|
-
}, /*#__PURE__*/React__default.default.createElement(Text.Text, {
|
|
253
|
-
size: "s"
|
|
254
|
-
}, content), /*#__PURE__*/React__default.default.createElement(StyledArrow, {
|
|
255
|
-
horizontalPlacement: position.horizontalPlacement,
|
|
256
|
-
verticalPlacement: position.verticalPlacement
|
|
257
|
-
})));
|
|
258
|
-
portal = /*#__PURE__*/ReactDOM.createPortal(tooltipElm, portalContainer || document.body);
|
|
259
|
-
}
|
|
260
119
|
const triggerElm = children ? /*#__PURE__*/React__default.default.cloneElement(children, {
|
|
261
120
|
ref: triggerRef,
|
|
262
121
|
...(isVisible && {
|
|
@@ -269,7 +128,24 @@ function Tooltip(_ref3) {
|
|
|
269
128
|
onBlur: handleTriggerBlur,
|
|
270
129
|
onKeyDown: handleTriggerKeyDown
|
|
271
130
|
}) : null;
|
|
272
|
-
|
|
131
|
+
const contentElm = /*#__PURE__*/React__default.default.createElement(Text.Text, {
|
|
132
|
+
size: "s"
|
|
133
|
+
}, content);
|
|
134
|
+
const tooltipElm = /*#__PURE__*/React__default.default.createElement(TooltipContent.TooltipContent, {
|
|
135
|
+
dataDSId: "Tooltip",
|
|
136
|
+
content: contentElm,
|
|
137
|
+
placement: placement,
|
|
138
|
+
portalContainer: portalContainer,
|
|
139
|
+
dataE2eTestId: dataE2eTestId,
|
|
140
|
+
isVisible: isVisible,
|
|
141
|
+
tooltipId: tooltipId,
|
|
142
|
+
triggerRef: triggerRef,
|
|
143
|
+
"aria-hidden": true,
|
|
144
|
+
role: "tooltip",
|
|
145
|
+
onTooltipPointerEnter: handleTooltipPointerEnter,
|
|
146
|
+
onTooltipPointerLeave: handleTooltipPointerLeave
|
|
147
|
+
});
|
|
148
|
+
return /*#__PURE__*/React__default.default.createElement(React__default.default.Fragment, null, triggerElm, tooltipElm);
|
|
273
149
|
}
|
|
274
150
|
|
|
275
151
|
exports.Tooltip = Tooltip;
|
|
@@ -0,0 +1,210 @@
|
|
|
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 _zindex = require('../../../build-tokens/_zindex.json.js');
|
|
11
|
+
var utils = require('./utils.js');
|
|
12
|
+
|
|
13
|
+
function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
|
|
14
|
+
|
|
15
|
+
var _styled__default = /*#__PURE__*/_interopDefault(_styled);
|
|
16
|
+
var React__default = /*#__PURE__*/_interopDefault(React);
|
|
17
|
+
|
|
18
|
+
const ANIMATION_DURATION = 200;
|
|
19
|
+
const SHOW_HIDE_DELAY = 200;
|
|
20
|
+
const StyledContainer = /*#__PURE__*/_styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
21
|
+
target: "e1k814k41"
|
|
22
|
+
} : {
|
|
23
|
+
target: "e1k814k41",
|
|
24
|
+
label: "StyledContainer"
|
|
25
|
+
})(_ref => {
|
|
26
|
+
let {
|
|
27
|
+
theme,
|
|
28
|
+
horizontalPlacement,
|
|
29
|
+
verticalPlacement,
|
|
30
|
+
contentPadding
|
|
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
|
+
const contentPaddingMap = {
|
|
40
|
+
s: theme.variables.size.spacing.xxs,
|
|
41
|
+
m: theme.variables.size.spacing.s
|
|
42
|
+
};
|
|
43
|
+
return {
|
|
44
|
+
position: "absolute",
|
|
45
|
+
zIndex: _zindex.default.tooltip.value,
|
|
46
|
+
opacity: 0,
|
|
47
|
+
animation: `${ANIMATION_DURATION}ms ease-out forwards ${animation}`,
|
|
48
|
+
borderRadius: theme.variables.size.borderRadius.xs,
|
|
49
|
+
backgroundColor: theme.values.color.background.primary.default,
|
|
50
|
+
maxWidth: "224px",
|
|
51
|
+
width: "initial",
|
|
52
|
+
boxSizing: "border-box",
|
|
53
|
+
padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,
|
|
54
|
+
...(contentPadding && {
|
|
55
|
+
padding: contentPaddingMap[contentPadding]
|
|
56
|
+
}),
|
|
57
|
+
...(horizontalPlacement === "center" && {
|
|
58
|
+
transform: "translate(-50%)"
|
|
59
|
+
})
|
|
60
|
+
};
|
|
61
|
+
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TooltipContent.tsx"],"names":[],"mappings":"AAkEwB","file":"TooltipContent.tsx","sourcesContent":["import React, {\n  useState,\n  useRef,\n  useLayoutEffect,\n  useEffect,\n  MutableRefObject,\n  ReactElement,\n  useCallback,\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  ARROW_SIZE,\n  ARROW_OFFSET,\n  ANIMATION_DISTANCE,\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  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\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<TooltipContentProps, \"contentPadding\"> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({ theme, horizontalPlacement, verticalPlacement, contentPadding }) => {\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    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      width: \"initial\",\n      boxSizing: \"border-box\",\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"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 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 Popover 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  onTooltipPointerEnter,\n  onTooltipPointerLeave,\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 calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(placement, triggerRef, tooltipRef, document, window)\n      );\n    }\n  }, [triggerRef, tooltipRef, document, window, placement]);\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    } 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    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  let portal = null;\n\n  if (isVisible) {\n    const tooltipElm = (\n      <SubThemeProvider name=\"inverted\">\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          tabIndex={tabIndex}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          contentPadding={contentPadding}\n          onPointerEnter={onTooltipPointerEnter}\n          onPointerLeave={onTooltipPointerLeave}\n        >\n          {content}\n          <StyledArrow\n            horizontalPlacement={style.horizontalPlacement}\n            verticalPlacement={style.verticalPlacement}\n          />\n        </StyledContainer>\n      </SubThemeProvider>\n    );\n\n    portal = createPortal(tooltipElm, portalContainer || document.body);\n  }\n\n  return portal;\n}\n"]} */");
|
|
62
|
+
const StyledArrow = /*#__PURE__*/_styled__default.default("div", process.env.NODE_ENV === "production" ? {
|
|
63
|
+
target: "e1k814k40"
|
|
64
|
+
} : {
|
|
65
|
+
target: "e1k814k40",
|
|
66
|
+
label: "StyledArrow"
|
|
67
|
+
})(_ref2 => {
|
|
68
|
+
let {
|
|
69
|
+
theme,
|
|
70
|
+
verticalPlacement,
|
|
71
|
+
horizontalPlacement
|
|
72
|
+
} = _ref2;
|
|
73
|
+
return {
|
|
74
|
+
position: "absolute",
|
|
75
|
+
width: 0,
|
|
76
|
+
height: 0,
|
|
77
|
+
borderLeft: `${utils.ARROW_SIZE}px solid transparent`,
|
|
78
|
+
borderRight: `${utils.ARROW_SIZE}px solid transparent`,
|
|
79
|
+
...(verticalPlacement === "top" && {
|
|
80
|
+
top: "100%",
|
|
81
|
+
borderTop: `${utils.ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`
|
|
82
|
+
}),
|
|
83
|
+
...(verticalPlacement === "bottom" && {
|
|
84
|
+
top: `-${utils.ARROW_SIZE}px`,
|
|
85
|
+
borderBottom: `${utils.ARROW_SIZE}px solid ${theme.values.color.background.primary.default}`
|
|
86
|
+
}),
|
|
87
|
+
...(horizontalPlacement === "center" && {
|
|
88
|
+
left: "50%",
|
|
89
|
+
transform: "translate(-50%)"
|
|
90
|
+
}),
|
|
91
|
+
...(horizontalPlacement === "right" && {
|
|
92
|
+
left: `${utils.ARROW_OFFSET}px`
|
|
93
|
+
}),
|
|
94
|
+
...(horizontalPlacement === "left" && {
|
|
95
|
+
right: `${utils.ARROW_OFFSET}px`
|
|
96
|
+
})
|
|
97
|
+
};
|
|
98
|
+
}, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TooltipContent.tsx"],"names":[],"mappings":"AAkHoB","file":"TooltipContent.tsx","sourcesContent":["import React, {\n  useState,\n  useRef,\n  useLayoutEffect,\n  useEffect,\n  MutableRefObject,\n  ReactElement,\n  useCallback,\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  ARROW_SIZE,\n  ARROW_OFFSET,\n  ANIMATION_DISTANCE,\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  onTooltipPointerEnter?: React.PointerEventHandler<HTMLDivElement>;\n  onTooltipPointerLeave?: React.PointerEventHandler<HTMLDivElement>;\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<TooltipContentProps, \"contentPadding\"> & {\n  horizontalPlacement: TooltipStyle[\"horizontalPlacement\"];\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n};\n\nconst ANIMATION_DURATION = 200;\nconst SHOW_HIDE_DELAY = 200;\n\nconst StyledContainer = styled.div<StyledContainerProps>(\n  ({ theme, horizontalPlacement, verticalPlacement, contentPadding }) => {\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    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      width: \"initial\",\n      boxSizing: \"border-box\",\n      padding: `${theme.variables.size.spacing.xs} ${theme.variables.size.spacing.s}`,\n\n      ...(contentPadding && {\n        padding: contentPaddingMap[contentPadding],\n      }),\n      ...(horizontalPlacement === \"center\" && {\n        transform: \"translate(-50%)\",\n      }),\n    };\n  }\n);\n\ntype StyledArrowProps = {\n  verticalPlacement: TooltipStyle[\"verticalPlacement\"];\n  horizontalPlacement: TooltipStyle[\"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 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 Popover 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  onTooltipPointerEnter,\n  onTooltipPointerLeave,\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 calculateStyle = useCallback(() => {\n    if (triggerRef.current && tooltipRef.current) {\n      // calculate tooltip style\n      setStyle(\n        getTooltipStyle(placement, triggerRef, tooltipRef, document, window)\n      );\n    }\n  }, [triggerRef, tooltipRef, document, window, placement]);\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    } 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    };\n  }, [isVisible, calculateStyle, window, tooltipRef]);\n\n  let portal = null;\n\n  if (isVisible) {\n    const tooltipElm = (\n      <SubThemeProvider name=\"inverted\">\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          tabIndex={tabIndex}\n          horizontalPlacement={style.horizontalPlacement}\n          verticalPlacement={style.verticalPlacement}\n          contentPadding={contentPadding}\n          onPointerEnter={onTooltipPointerEnter}\n          onPointerLeave={onTooltipPointerLeave}\n        >\n          {content}\n          <StyledArrow\n            horizontalPlacement={style.horizontalPlacement}\n            verticalPlacement={style.verticalPlacement}\n          />\n        </StyledContainer>\n      </SubThemeProvider>\n    );\n\n    portal = createPortal(tooltipElm, portalContainer || document.body);\n  }\n\n  return portal;\n}\n"]} */");
|
|
99
|
+
const initialStyle = {
|
|
100
|
+
top: 0,
|
|
101
|
+
left: 0,
|
|
102
|
+
verticalPlacement: "top",
|
|
103
|
+
horizontalPlacement: "center"
|
|
104
|
+
};
|
|
105
|
+
let lastTooltipHideTimestamp = 0;
|
|
106
|
+
|
|
107
|
+
/* Disable animation if time between last close and new open is less than 500ms + SHOW_HIDE_DELAY */
|
|
108
|
+
function getAnimationDuration() {
|
|
109
|
+
let animationDuration = `${ANIMATION_DURATION}ms`;
|
|
110
|
+
if (lastTooltipHideTimestamp) {
|
|
111
|
+
const timeSinceLastTooltip = Date.now() - lastTooltipHideTimestamp;
|
|
112
|
+
if (timeSinceLastTooltip < 500 + SHOW_HIDE_DELAY) {
|
|
113
|
+
animationDuration = "0ms";
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
return animationDuration;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/** This component is used to display the overlay for both Popover and Tooltip components */
|
|
120
|
+
function TooltipContent(_ref3) {
|
|
121
|
+
let {
|
|
122
|
+
placement = "auto",
|
|
123
|
+
content,
|
|
124
|
+
tooltipId,
|
|
125
|
+
triggerRef,
|
|
126
|
+
portalContainer,
|
|
127
|
+
dataE2eTestId,
|
|
128
|
+
dataDSId,
|
|
129
|
+
isVisible,
|
|
130
|
+
"aria-hidden": ariaHidden,
|
|
131
|
+
role,
|
|
132
|
+
tabIndex,
|
|
133
|
+
contentPadding,
|
|
134
|
+
onTooltipPointerEnter,
|
|
135
|
+
onTooltipPointerLeave
|
|
136
|
+
} = _ref3;
|
|
137
|
+
const [style, setStyle] = React.useState(initialStyle);
|
|
138
|
+
const tooltipRef = React.useRef(null);
|
|
139
|
+
const document = useDocument.useDocument();
|
|
140
|
+
const window = useWindow.useWindow();
|
|
141
|
+
const calculateStyle = React.useCallback(() => {
|
|
142
|
+
if (triggerRef.current && tooltipRef.current) {
|
|
143
|
+
// calculate tooltip style
|
|
144
|
+
setStyle(utils.getTooltipStyle(placement, triggerRef, tooltipRef, document, window));
|
|
145
|
+
}
|
|
146
|
+
}, [triggerRef, tooltipRef, document, window, placement]);
|
|
147
|
+
|
|
148
|
+
// This layout effect to re-render with updated position after determining content width
|
|
149
|
+
React.useLayoutEffect(() => {
|
|
150
|
+
if (isVisible) {
|
|
151
|
+
calculateStyle();
|
|
152
|
+
}
|
|
153
|
+
}, [isVisible, calculateStyle, contentPadding, content]);
|
|
154
|
+
|
|
155
|
+
// Re-position tooltip if it moves out of the viewport by 10% and on window resize
|
|
156
|
+
React.useEffect(() => {
|
|
157
|
+
let observer;
|
|
158
|
+
if (typeof IntersectionObserver !== "undefined" && isVisible && tooltipRef.current) {
|
|
159
|
+
observer = new IntersectionObserver(entries => {
|
|
160
|
+
entries.forEach(() => {
|
|
161
|
+
calculateStyle();
|
|
162
|
+
});
|
|
163
|
+
}, {
|
|
164
|
+
threshold: 0.9
|
|
165
|
+
});
|
|
166
|
+
observer.observe(tooltipRef.current);
|
|
167
|
+
window.addEventListener("resize", calculateStyle);
|
|
168
|
+
} else if (!isVisible) {
|
|
169
|
+
// log time when tooltip closes
|
|
170
|
+
lastTooltipHideTimestamp = Date.now();
|
|
171
|
+
}
|
|
172
|
+
return () => {
|
|
173
|
+
if (observer) {
|
|
174
|
+
observer.disconnect();
|
|
175
|
+
}
|
|
176
|
+
window.removeEventListener("resize", calculateStyle);
|
|
177
|
+
};
|
|
178
|
+
}, [isVisible, calculateStyle, window, tooltipRef]);
|
|
179
|
+
let portal = null;
|
|
180
|
+
if (isVisible) {
|
|
181
|
+
const tooltipElm = /*#__PURE__*/React__default.default.createElement(SubThemeProvider.SubThemeProvider, {
|
|
182
|
+
name: "inverted"
|
|
183
|
+
}, /*#__PURE__*/React__default.default.createElement(StyledContainer, {
|
|
184
|
+
"data-e2e-test-id": dataE2eTestId,
|
|
185
|
+
"data-ds-id": dataDSId,
|
|
186
|
+
style: {
|
|
187
|
+
top: style.top,
|
|
188
|
+
left: style.left,
|
|
189
|
+
animationDuration: getAnimationDuration()
|
|
190
|
+
},
|
|
191
|
+
ref: tooltipRef,
|
|
192
|
+
id: tooltipId,
|
|
193
|
+
role: role,
|
|
194
|
+
"aria-hidden": ariaHidden,
|
|
195
|
+
tabIndex: tabIndex,
|
|
196
|
+
horizontalPlacement: style.horizontalPlacement,
|
|
197
|
+
verticalPlacement: style.verticalPlacement,
|
|
198
|
+
contentPadding: contentPadding,
|
|
199
|
+
onPointerEnter: onTooltipPointerEnter,
|
|
200
|
+
onPointerLeave: onTooltipPointerLeave
|
|
201
|
+
}, content, /*#__PURE__*/React__default.default.createElement(StyledArrow, {
|
|
202
|
+
horizontalPlacement: style.horizontalPlacement,
|
|
203
|
+
verticalPlacement: style.verticalPlacement
|
|
204
|
+
})));
|
|
205
|
+
portal = /*#__PURE__*/ReactDOM.createPortal(tooltipElm, portalContainer || document.body);
|
|
206
|
+
}
|
|
207
|
+
return portal;
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
exports.TooltipContent = TooltipContent;
|
|
@@ -7,9 +7,9 @@ const TOOLTIP_MARGIN = ARROW_SIZE + ANIMATION_DISTANCE + DISTANCE_FROM_TRIGGER;
|
|
|
7
7
|
const ARROW_OFFSET = 8;
|
|
8
8
|
|
|
9
9
|
/**
|
|
10
|
-
* Get tooltip position
|
|
10
|
+
* Get tooltip position and width
|
|
11
11
|
*/
|
|
12
|
-
function
|
|
12
|
+
function getTooltipStyle(placement, triggerRef, tooltipRef, document, window) {
|
|
13
13
|
const triggerRect = triggerRef.current.getBoundingClientRect();
|
|
14
14
|
const tooltipRect = tooltipRef.current.getBoundingClientRect();
|
|
15
15
|
const viewportWidth = document.documentElement.clientWidth;
|
|
@@ -78,4 +78,4 @@ exports.ARROW_OFFSET = ARROW_OFFSET;
|
|
|
78
78
|
exports.ARROW_SIZE = ARROW_SIZE;
|
|
79
79
|
exports.DISTANCE_FROM_TRIGGER = DISTANCE_FROM_TRIGGER;
|
|
80
80
|
exports.TOOLTIP_MARGIN = TOOLTIP_MARGIN;
|
|
81
|
-
exports.
|
|
81
|
+
exports.getTooltipStyle = getTooltipStyle;
|