Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

@ndla/tooltip

Package Overview
Dependencies
Maintainers
5
Versions
168
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@ndla/tooltip - npm Package Compare versions

Comparing version 0.2.11 to 0.2.12

94

es/Tooltip.js

@@ -44,3 +44,3 @@ import _styled from "@emotion/styled-base";

styles: "position:relative;",
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAeiC","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAeiC","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  position: absolute;\n  z-index: 9999;\n  pointer-events: none;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    const { align } = this.props;\n    if (this.state.showTooltip) {\n      const widthRef = this.focusableChild.offsetWidth;\n      const heightRef = this.focusableChild.offsetHeight;\n      const elementRect = this.focusableChild.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        align === 'top' ||\n        align === 'bottom' ||\n        (align === 'left' && leftRef - tooltipWidth < 20) ||\n        (align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  componentDidMount() {\n    this.focusableChild = this.contentRef.current.querySelector(\n      'a, button, [role=\"button\"]',\n    );\n    if (this.focusableChild) {\n      this.focusableChild.addEventListener('focusin', this.handleShowTooltip);\n      this.focusableChild.addEventListener('focusout', this.handleHideTooltip);\n    }\n  }\n\n  componentWillUnmount() {\n    if (this.focusableChild) {\n      this.focusableChild.removeEventListener(\n        'focusin',\n        this.handleShowTooltip,\n      );\n      this.focusableChild.removeEventListener(\n        'focusout',\n        this.handleHideTooltip,\n      );\n    }\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  render() {\n    const {\n      tooltipContainerClass,\n      className,\n      delay,\n      tooltip,\n      children,\n    } = this.props;\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={tooltipContainerClass}>\n          <span className={className}>{children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {tooltip}\n          </span>\n        </Fade>\n        <div\n          aria-label={tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseLeave={this.handleHideTooltip}\n          className={className}>\n          {children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"
});

@@ -50,11 +50,3 @@

/*#__PURE__*/
css("display:block;color:#fff;position:absolute;z-index:9999;background:", colors.brand.primary, ";border-radius:2px;padding:", spacing.xsmall, " ", spacing.small, ";font-family:", fonts.sans, ";", fonts.sizes(14, 1.2), " font-weight:", fonts.weight.normal, ";color:$white;text-align:center;white-space:nowrap;max-width:calc(100vw - #{", spacing.normal, "});pointer-events:none;label:tooltipCss;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAmBsB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));
var contentCSS = process.env.NODE_ENV === "production" ? {
name: "1hji32s-contentCSS",
styles: "display:inline-block;label:contentCSS;"
} : {
name: "1hji32s-contentCSS",
styles: "display:inline-block;label:contentCSS;",
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAoCsB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"
};
css("display:block;color:#fff;background:", colors.brand.primary, ";border-radius:2px;padding:", spacing.xsmall, " ", spacing.small, ";font-family:", fonts.sans, ";", fonts.sizes(14, 1.2), " font-weight:", fonts.weight.normal, ";color:$white;text-align:center;white-space:nowrap;max-width:calc(100vw - #{", spacing.normal, "});label:tooltipCss;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAmBsB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  position: absolute;\n  z-index: 9999;\n  pointer-events: none;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    const { align } = this.props;\n    if (this.state.showTooltip) {\n      const widthRef = this.focusableChild.offsetWidth;\n      const heightRef = this.focusableChild.offsetHeight;\n      const elementRect = this.focusableChild.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        align === 'top' ||\n        align === 'bottom' ||\n        (align === 'left' && leftRef - tooltipWidth < 20) ||\n        (align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  componentDidMount() {\n    this.focusableChild = this.contentRef.current.querySelector(\n      'a, button, [role=\"button\"]',\n    );\n    if (this.focusableChild) {\n      this.focusableChild.addEventListener('focusin', this.handleShowTooltip);\n      this.focusableChild.addEventListener('focusout', this.handleHideTooltip);\n    }\n  }\n\n  componentWillUnmount() {\n    if (this.focusableChild) {\n      this.focusableChild.removeEventListener(\n        'focusin',\n        this.handleShowTooltip,\n      );\n      this.focusableChild.removeEventListener(\n        'focusout',\n        this.handleHideTooltip,\n      );\n    }\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  render() {\n    const {\n      tooltipContainerClass,\n      className,\n      delay,\n      tooltip,\n      children,\n    } = this.props;\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={tooltipContainerClass}>\n          <span className={className}>{children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {tooltip}\n          </span>\n        </Fade>\n        <div\n          aria-label={tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseLeave={this.handleHideTooltip}\n          className={className}>\n          {children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));

@@ -64,7 +56,7 @@ var Fade = _styled("div", {

label: "Fade"
})("opacity:0;@keyframes fadeInTooltip{0%{opacity:0;}100%{opacity:1;}}@keyframes fadeOutTooltip{0%{opacity:1;}100%{opacity:0;}}animation-fill-mode:forwards;animation-delay:", function (props) {
})("opacity:0;position:absolute;z-index:9999;pointer-events:none;@keyframes fadeInTooltip{0%{opacity:0;}100%{opacity:1;}}@keyframes fadeOutTooltip{0%{opacity:1;}100%{opacity:0;}}animation-fill-mode:forwards;animation-delay:", function (props) {
return props.delay;
}, "ms;animation-name:", function (props) {
return props.animateIn ? 'fadeInTooltip' : '';
}, ";animation-duration:300ms;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAwCuB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));
}, ";animation-duration:300ms;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAiCuB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  position: absolute;\n  z-index: 9999;\n  pointer-events: none;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    const { align } = this.props;\n    if (this.state.showTooltip) {\n      const widthRef = this.focusableChild.offsetWidth;\n      const heightRef = this.focusableChild.offsetHeight;\n      const elementRect = this.focusableChild.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        align === 'top' ||\n        align === 'bottom' ||\n        (align === 'left' && leftRef - tooltipWidth < 20) ||\n        (align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  componentDidMount() {\n    this.focusableChild = this.contentRef.current.querySelector(\n      'a, button, [role=\"button\"]',\n    );\n    if (this.focusableChild) {\n      this.focusableChild.addEventListener('focusin', this.handleShowTooltip);\n      this.focusableChild.addEventListener('focusout', this.handleHideTooltip);\n    }\n  }\n\n  componentWillUnmount() {\n    if (this.focusableChild) {\n      this.focusableChild.removeEventListener(\n        'focusin',\n        this.handleShowTooltip,\n      );\n      this.focusableChild.removeEventListener(\n        'focusout',\n        this.handleHideTooltip,\n      );\n    }\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  render() {\n    const {\n      tooltipContainerClass,\n      className,\n      delay,\n      tooltip,\n      children,\n    } = this.props;\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={tooltipContainerClass}>\n          <span className={className}>{children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {tooltip}\n          </span>\n        </Fade>\n        <div\n          aria-label={tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseLeave={this.handleHideTooltip}\n          className={className}>\n          {children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));

@@ -87,3 +79,2 @@ var Tooltip =

_this.handleHideTooltip = _this.handleHideTooltip.bind(_assertThisInitialized(_this));
_this.handleKeyPress = _this.handleKeyPress.bind(_assertThisInitialized(_this));
_this.contentRef = React.createRef();

@@ -98,7 +89,8 @@ _this.tooltipRef = React.createRef();

var currentStyles = {};
var align = this.props.align;
if (this.state.showTooltip) {
var widthRef = this.contentRef.current.offsetWidth;
var heightRef = this.contentRef.current.offsetHeight;
var elementRect = this.contentRef.current.getBoundingClientRect();
var widthRef = this.focusableChild.offsetWidth;
var heightRef = this.focusableChild.offsetHeight;
var elementRect = this.focusableChild.getBoundingClientRect();
var leftRef = elementRect.left;

@@ -111,3 +103,3 @@ var tooltipWidth = this.tooltipRef.current.offsetWidth;

currentStyles.top = "-".concat(this.tooltipRef.current.offsetHeight + 10, "px");
} else if (this.props.align === 'top' || this.props.align === 'bottom' || this.props.align === 'left' && leftRef - tooltipWidth < 20 || this.props.align === 'right' && leftRef + widthRef + tooltipWidth > window.innerWidth - 40) {
} else if (align === 'top' || align === 'bottom' || align === 'left' && leftRef - tooltipWidth < 20 || align === 'right' && leftRef + widthRef + tooltipWidth > window.innerWidth - 40) {
var centeredLeft = leftRef + widthRef / 2;

@@ -120,3 +112,3 @@ var moveHorizontal = Math.max(centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth, 0);

if (this.props.align === 'bottom') {
if (align === 'bottom') {
currentStyles.transform = "translate(calc(-50% + ".concat(widthRef / 2 - moveHorizontal, "px), calc(").concat(heightRef, "px + ").concat(spacing.xsmall, "))");

@@ -126,3 +118,3 @@ } else {

}
} else if (this.props.align === 'left') {
} else if (align === 'left') {
currentStyles.transform = "translate(calc(-100% - ".concat(spacing.xsmall, "), calc(-50% + ").concat(heightRef / 2, "px))");

@@ -137,2 +129,20 @@ } else {

}, {
key: "componentDidMount",
value: function componentDidMount() {
this.focusableChild = this.contentRef.current.querySelector('a, button, [role="button"]');
if (this.focusableChild) {
this.focusableChild.addEventListener('focusin', this.handleShowTooltip);
this.focusableChild.addEventListener('focusout', this.handleHideTooltip);
}
}
}, {
key: "componentWillUnmount",
value: function componentWillUnmount() {
if (this.focusableChild) {
this.focusableChild.removeEventListener('focusin', this.handleShowTooltip);
this.focusableChild.removeEventListener('focusout', this.handleHideTooltip);
}
}
}, {
key: "handleShowTooltip",

@@ -152,29 +162,24 @@ value: function handleShowTooltip() {

}, {
key: "handleKeyPress",
value: function handleKeyPress(e) {
if (e.key === 'Enter') {
try {
this.contentRef.current.querySelectorAll('[type="button"], a')[0].click();
} catch (err) {
console.log('error', err); // eslint-disable-line no-console
}
}
}
}, {
key: "render",
value: function render() {
// If phone ignore all tooltips //
var _this$props = this.props,
tooltipContainerClass = _this$props.tooltipContainerClass,
className = _this$props.className,
delay = _this$props.delay,
tooltip = _this$props.tooltip,
children = _this$props.children; // If phone ignore all tooltips //
if (isMobile) {
return ___EmotionJSX("div", {
className: this.props.tooltipContainerClass
className: tooltipContainerClass
}, ___EmotionJSX("span", {
className: this.props.className
}, this.props.children));
className: className
}, children));
}
return ___EmotionJSX(TooltipWrapper, {
className: this.props.tooltipContainerClass
className: tooltipContainerClass
}, ___EmotionJSX(Fade, {
animateIn: this.state.showTooltip,
delay: this.props.delay
delay: delay
}, ___EmotionJSX("span", {

@@ -185,16 +190,9 @@ role: "tooltip",

ref: this.tooltipRef
}, this.props.tooltip)), ___EmotionJSX("div", {
role: "button",
tabIndex: 0,
"aria-label": this.props.tooltip,
}, tooltip)), ___EmotionJSX("div", {
"aria-label": tooltip,
ref: this.contentRef,
onMouseEnter: this.handleShowTooltip,
onMouseOut: this.handleHideTooltip,
onMouseMove: this.handleShowTooltip,
onFocus: this.handleShowTooltip,
onKeyPress: this.handleKeyPress,
onBlur: this.handleHideTooltip,
css: contentCSS,
className: this.props.className
}, this.props.children));
onMouseLeave: this.handleHideTooltip,
className: className
}, children));
}

@@ -201,0 +199,0 @@ }]);

@@ -51,23 +51,15 @@ "use strict";

styles: "position:relative;",
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAeiC","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAeiC","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  position: absolute;\n  z-index: 9999;\n  pointer-events: none;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    const { align } = this.props;\n    if (this.state.showTooltip) {\n      const widthRef = this.focusableChild.offsetWidth;\n      const heightRef = this.focusableChild.offsetHeight;\n      const elementRect = this.focusableChild.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        align === 'top' ||\n        align === 'bottom' ||\n        (align === 'left' && leftRef - tooltipWidth < 20) ||\n        (align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  componentDidMount() {\n    this.focusableChild = this.contentRef.current.querySelector(\n      'a, button, [role=\"button\"]',\n    );\n    if (this.focusableChild) {\n      this.focusableChild.addEventListener('focusin', this.handleShowTooltip);\n      this.focusableChild.addEventListener('focusout', this.handleHideTooltip);\n    }\n  }\n\n  componentWillUnmount() {\n    if (this.focusableChild) {\n      this.focusableChild.removeEventListener(\n        'focusin',\n        this.handleShowTooltip,\n      );\n      this.focusableChild.removeEventListener(\n        'focusout',\n        this.handleHideTooltip,\n      );\n    }\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  render() {\n    const {\n      tooltipContainerClass,\n      className,\n      delay,\n      tooltip,\n      children,\n    } = this.props;\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={tooltipContainerClass}>\n          <span className={className}>{children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {tooltip}\n          </span>\n        </Fade>\n        <div\n          aria-label={tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseLeave={this.handleHideTooltip}\n          className={className}>\n          {children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"
});
var tooltipCss =
/*#__PURE__*/
(0, _core.css)("display:block;color:#fff;position:absolute;z-index:9999;background:", _core2.colors.brand.primary, ";border-radius:2px;padding:", _core2.spacing.xsmall, " ", _core2.spacing.small, ";font-family:", _core2.fonts.sans, ";", _core2.fonts.sizes(14, 1.2), " font-weight:", _core2.fonts.weight.normal, ";color:$white;text-align:center;white-space:nowrap;max-width:calc(100vw - #{", _core2.spacing.normal, "});pointer-events:none;label:tooltipCss;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAmBsB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));
var contentCSS = process.env.NODE_ENV === "production" ? {
name: "1hji32s-contentCSS",
styles: "display:inline-block;label:contentCSS;"
} : {
name: "1hji32s-contentCSS",
styles: "display:inline-block;label:contentCSS;",
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAoCsB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"
};
(0, _core.css)("display:block;color:#fff;background:", _core2.colors.brand.primary, ";border-radius:2px;padding:", _core2.spacing.xsmall, " ", _core2.spacing.small, ";font-family:", _core2.fonts.sans, ";", _core2.fonts.sizes(14, 1.2), " font-weight:", _core2.fonts.weight.normal, ";color:$white;text-align:center;white-space:nowrap;max-width:calc(100vw - #{", _core2.spacing.normal, "});label:tooltipCss;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAmBsB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  position: absolute;\n  z-index: 9999;\n  pointer-events: none;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    const { align } = this.props;\n    if (this.state.showTooltip) {\n      const widthRef = this.focusableChild.offsetWidth;\n      const heightRef = this.focusableChild.offsetHeight;\n      const elementRect = this.focusableChild.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        align === 'top' ||\n        align === 'bottom' ||\n        (align === 'left' && leftRef - tooltipWidth < 20) ||\n        (align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  componentDidMount() {\n    this.focusableChild = this.contentRef.current.querySelector(\n      'a, button, [role=\"button\"]',\n    );\n    if (this.focusableChild) {\n      this.focusableChild.addEventListener('focusin', this.handleShowTooltip);\n      this.focusableChild.addEventListener('focusout', this.handleHideTooltip);\n    }\n  }\n\n  componentWillUnmount() {\n    if (this.focusableChild) {\n      this.focusableChild.removeEventListener(\n        'focusin',\n        this.handleShowTooltip,\n      );\n      this.focusableChild.removeEventListener(\n        'focusout',\n        this.handleHideTooltip,\n      );\n    }\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  render() {\n    const {\n      tooltipContainerClass,\n      className,\n      delay,\n      tooltip,\n      children,\n    } = this.props;\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={tooltipContainerClass}>\n          <span className={className}>{children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {tooltip}\n          </span>\n        </Fade>\n        <div\n          aria-label={tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseLeave={this.handleHideTooltip}\n          className={className}>\n          {children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));
var Fade = (0, _styledBase.default)("div", {
target: "exsy4zb1",
label: "Fade"
})("opacity:0;@keyframes fadeInTooltip{0%{opacity:0;}100%{opacity:1;}}@keyframes fadeOutTooltip{0%{opacity:1;}100%{opacity:0;}}animation-fill-mode:forwards;animation-delay:", function (props) {
})("opacity:0;position:absolute;z-index:9999;pointer-events:none;@keyframes fadeInTooltip{0%{opacity:0;}100%{opacity:1;}}@keyframes fadeOutTooltip{0%{opacity:1;}100%{opacity:0;}}animation-fill-mode:forwards;animation-delay:", function (props) {
return props.delay;
}, "ms;animation-name:", function (props) {
return props.animateIn ? 'fadeInTooltip' : '';
}, ";animation-duration:300ms;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAwCuB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  position: absolute;\n  z-index: 9999;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n  pointer-events: none;\n`;\n\nconst contentCSS = css`\n  display: inline-block;\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.handleKeyPress = this.handleKeyPress.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    if (this.state.showTooltip) {\n      const widthRef = this.contentRef.current.offsetWidth;\n      const heightRef = this.contentRef.current.offsetHeight;\n      const elementRect = this.contentRef.current.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        this.props.align === 'top' ||\n        this.props.align === 'bottom' ||\n        (this.props.align === 'left' && leftRef - tooltipWidth < 20) ||\n        (this.props.align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (this.props.align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (this.props.align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  handleKeyPress(e) {\n    if (e.key === 'Enter') {\n      try {\n        this.contentRef.current\n          .querySelectorAll('[type=\"button\"], a')[0]\n          .click();\n      } catch (err) {\n        console.log('error', err); // eslint-disable-line no-console\n      }\n    }\n  }\n\n  render() {\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={this.props.tooltipContainerClass}>\n          <span className={this.props.className}>{this.props.children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={this.props.tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={this.props.delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {this.props.tooltip}\n          </span>\n        </Fade>\n        <div\n          role=\"button\"\n          tabIndex={0}\n          aria-label={this.props.tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseOut={this.handleHideTooltip}\n          onMouseMove={this.handleShowTooltip}\n          onFocus={this.handleShowTooltip}\n          onKeyPress={this.handleKeyPress}\n          onBlur={this.handleHideTooltip}\n          css={contentCSS}\n          className={this.props.className}>\n          {this.props.children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));
}, ";animation-duration:300ms;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Tooltip.jsx"],"names":[],"mappings":"AAiCuB","file":"Tooltip.jsx","sourcesContent":["/**\n * Copyright (c) 2016-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { Component } from 'react';\nimport PropTypes from 'prop-types';\nimport { isMobile, isIE } from 'react-device-detect';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/core';\nimport { spacing, colors, fonts } from '@ndla/core';\n\nconst TooltipWrapper = styled.div`\n  position: relative;\n`;\n\nconst tooltipCss = css`\n  display: block;\n  color: #fff;\n  background: ${colors.brand.primary};\n  border-radius: 2px;\n  padding: ${spacing.xsmall} ${spacing.small};\n  font-family: ${fonts.sans};\n  ${fonts.sizes(14, 1.2)} font-weight: ${fonts.weight.normal};\n  color: $white;\n  text-align: center;\n  white-space: nowrap;\n  max-width: calc(100vw - #{${spacing.normal}});\n`;\n\nconst Fade = styled.div`\n  opacity: 0;\n  position: absolute;\n  z-index: 9999;\n  pointer-events: none;\n  @keyframes fadeInTooltip {\n    0% {\n      opacity: 0;\n    }\n    100% {\n      opacity: 1;\n    }\n  }\n  @keyframes fadeOutTooltip {\n    0% {\n      opacity: 1;\n    }\n    100% {\n      opacity: 0;\n    }\n  }\n  animation-fill-mode: forwards;\n  animation-delay: ${props => props.delay}ms;\n  animation-name: ${props => (props.animateIn ? 'fadeInTooltip' : '')};\n  animation-duration: 300ms;\n`;\n\nclass Tooltip extends Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      showTooltip: false,\n    };\n    this.handleShowTooltip = this.handleShowTooltip.bind(this);\n    this.handleHideTooltip = this.handleHideTooltip.bind(this);\n    this.contentRef = React.createRef();\n    this.tooltipRef = React.createRef();\n  }\n\n  getPosition() {\n    const currentStyles = {};\n    const { align } = this.props;\n    if (this.state.showTooltip) {\n      const widthRef = this.focusableChild.offsetWidth;\n      const heightRef = this.focusableChild.offsetHeight;\n      const elementRect = this.focusableChild.getBoundingClientRect();\n      const leftRef = elementRect.left;\n      const tooltipWidth = this.tooltipRef.current.offsetWidth;\n\n      if (isIE) {\n        // IE is bad with transform % + px..\n        currentStyles.left = `-${(this.tooltipRef.current.offsetWidth -\n          widthRef) /\n          2}px`;\n        currentStyles.top = `-${this.tooltipRef.current.offsetHeight + 10}px`;\n      } else if (\n        align === 'top' ||\n        align === 'bottom' ||\n        (align === 'left' && leftRef - tooltipWidth < 20) ||\n        (align === 'right' &&\n          leftRef + widthRef + tooltipWidth > window.innerWidth - 40)\n      ) {\n        const centeredLeft = leftRef + widthRef / 2;\n        let moveHorizontal = Math.max(\n          centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth,\n          0,\n        );\n        if (moveHorizontal === 0) {\n          moveHorizontal = Math.min(-(tooltipWidth / 2 - centeredLeft + 20), 0);\n        }\n        if (align === 'bottom') {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(${heightRef}px + ${spacing.xsmall}))`;\n        } else {\n          currentStyles.transform = `translate(calc(-50% + ${widthRef / 2 -\n            moveHorizontal}px), calc(-100% - ${spacing.xsmall}))`;\n        }\n      } else if (align === 'left') {\n        currentStyles.transform = `translate(calc(-100% - ${\n          spacing.xsmall\n        }), calc(-50% + ${heightRef / 2}px))`;\n      } else {\n        currentStyles.transform = `translate(calc(${widthRef}px + 0.25rem), calc(-50% + ${heightRef /\n          2}px))`;\n      }\n    }\n\n    return currentStyles;\n  }\n\n  componentDidMount() {\n    this.focusableChild = this.contentRef.current.querySelector(\n      'a, button, [role=\"button\"]',\n    );\n    if (this.focusableChild) {\n      this.focusableChild.addEventListener('focusin', this.handleShowTooltip);\n      this.focusableChild.addEventListener('focusout', this.handleHideTooltip);\n    }\n  }\n\n  componentWillUnmount() {\n    if (this.focusableChild) {\n      this.focusableChild.removeEventListener(\n        'focusin',\n        this.handleShowTooltip,\n      );\n      this.focusableChild.removeEventListener(\n        'focusout',\n        this.handleHideTooltip,\n      );\n    }\n  }\n\n  handleShowTooltip() {\n    this.setState({ showTooltip: !this.props.disabled });\n  }\n\n  handleHideTooltip() {\n    this.setState({ showTooltip: false });\n  }\n\n  render() {\n    const {\n      tooltipContainerClass,\n      className,\n      delay,\n      tooltip,\n      children,\n    } = this.props;\n    // If phone ignore all tooltips //\n    if (isMobile) {\n      return (\n        <div className={tooltipContainerClass}>\n          <span className={className}>{children}</span>\n        </div>\n      );\n    }\n\n    return (\n      <TooltipWrapper className={tooltipContainerClass}>\n        <Fade animateIn={this.state.showTooltip} delay={delay}>\n          <span\n            role=\"tooltip\"\n            css={tooltipCss}\n            style={this.getPosition()}\n            ref={this.tooltipRef}>\n            {tooltip}\n          </span>\n        </Fade>\n        <div\n          aria-label={tooltip}\n          ref={this.contentRef}\n          onMouseEnter={this.handleShowTooltip}\n          onMouseLeave={this.handleHideTooltip}\n          className={className}>\n          {children}\n        </div>\n      </TooltipWrapper>\n    );\n  }\n}\n\nTooltip.propTypes = {\n  children: PropTypes.node.isRequired,\n  tooltip: PropTypes.string.isRequired,\n  delay: PropTypes.number,\n  disabled: PropTypes.bool,\n  align: PropTypes.oneOf(['left', 'right', 'top', 'bottom']),\n  className: PropTypes.string,\n  tooltipContainerClass: PropTypes.string,\n};\n\nTooltip.defaultProps = {\n  align: 'top',\n  disabled: false,\n  delay: 0,\n  className: '',\n  tooltipContainerClass: '',\n};\n\nexport default Tooltip;\n"]} */"));

@@ -90,3 +82,2 @@ var Tooltip =

_this.handleHideTooltip = _this.handleHideTooltip.bind(_assertThisInitialized(_this));
_this.handleKeyPress = _this.handleKeyPress.bind(_assertThisInitialized(_this));
_this.contentRef = _react.default.createRef();

@@ -101,7 +92,8 @@ _this.tooltipRef = _react.default.createRef();

var currentStyles = {};
var align = this.props.align;
if (this.state.showTooltip) {
var widthRef = this.contentRef.current.offsetWidth;
var heightRef = this.contentRef.current.offsetHeight;
var elementRect = this.contentRef.current.getBoundingClientRect();
var widthRef = this.focusableChild.offsetWidth;
var heightRef = this.focusableChild.offsetHeight;
var elementRect = this.focusableChild.getBoundingClientRect();
var leftRef = elementRect.left;

@@ -114,3 +106,3 @@ var tooltipWidth = this.tooltipRef.current.offsetWidth;

currentStyles.top = "-".concat(this.tooltipRef.current.offsetHeight + 10, "px");
} else if (this.props.align === 'top' || this.props.align === 'bottom' || this.props.align === 'left' && leftRef - tooltipWidth < 20 || this.props.align === 'right' && leftRef + widthRef + tooltipWidth > window.innerWidth - 40) {
} else if (align === 'top' || align === 'bottom' || align === 'left' && leftRef - tooltipWidth < 20 || align === 'right' && leftRef + widthRef + tooltipWidth > window.innerWidth - 40) {
var centeredLeft = leftRef + widthRef / 2;

@@ -123,3 +115,3 @@ var moveHorizontal = Math.max(centeredLeft + tooltipWidth / 2 + 20 - window.innerWidth, 0);

if (this.props.align === 'bottom') {
if (align === 'bottom') {
currentStyles.transform = "translate(calc(-50% + ".concat(widthRef / 2 - moveHorizontal, "px), calc(").concat(heightRef, "px + ").concat(_core2.spacing.xsmall, "))");

@@ -129,3 +121,3 @@ } else {

}
} else if (this.props.align === 'left') {
} else if (align === 'left') {
currentStyles.transform = "translate(calc(-100% - ".concat(_core2.spacing.xsmall, "), calc(-50% + ").concat(heightRef / 2, "px))");

@@ -140,2 +132,20 @@ } else {

}, {
key: "componentDidMount",
value: function componentDidMount() {
this.focusableChild = this.contentRef.current.querySelector('a, button, [role="button"]');
if (this.focusableChild) {
this.focusableChild.addEventListener('focusin', this.handleShowTooltip);
this.focusableChild.addEventListener('focusout', this.handleHideTooltip);
}
}
}, {
key: "componentWillUnmount",
value: function componentWillUnmount() {
if (this.focusableChild) {
this.focusableChild.removeEventListener('focusin', this.handleShowTooltip);
this.focusableChild.removeEventListener('focusout', this.handleHideTooltip);
}
}
}, {
key: "handleShowTooltip",

@@ -155,29 +165,24 @@ value: function handleShowTooltip() {

}, {
key: "handleKeyPress",
value: function handleKeyPress(e) {
if (e.key === 'Enter') {
try {
this.contentRef.current.querySelectorAll('[type="button"], a')[0].click();
} catch (err) {
console.log('error', err); // eslint-disable-line no-console
}
}
}
}, {
key: "render",
value: function render() {
// If phone ignore all tooltips //
var _this$props = this.props,
tooltipContainerClass = _this$props.tooltipContainerClass,
className = _this$props.className,
delay = _this$props.delay,
tooltip = _this$props.tooltip,
children = _this$props.children; // If phone ignore all tooltips //
if (_reactDeviceDetect.isMobile) {
return (0, _core.jsx)("div", {
className: this.props.tooltipContainerClass
className: tooltipContainerClass
}, (0, _core.jsx)("span", {
className: this.props.className
}, this.props.children));
className: className
}, children));
}
return (0, _core.jsx)(TooltipWrapper, {
className: this.props.tooltipContainerClass
className: tooltipContainerClass
}, (0, _core.jsx)(Fade, {
animateIn: this.state.showTooltip,
delay: this.props.delay
delay: delay
}, (0, _core.jsx)("span", {

@@ -188,16 +193,9 @@ role: "tooltip",

ref: this.tooltipRef
}, this.props.tooltip)), (0, _core.jsx)("div", {
role: "button",
tabIndex: 0,
"aria-label": this.props.tooltip,
}, tooltip)), (0, _core.jsx)("div", {
"aria-label": tooltip,
ref: this.contentRef,
onMouseEnter: this.handleShowTooltip,
onMouseOut: this.handleHideTooltip,
onMouseMove: this.handleShowTooltip,
onFocus: this.handleShowTooltip,
onKeyPress: this.handleKeyPress,
onBlur: this.handleHideTooltip,
css: contentCSS,
className: this.props.className
}, this.props.children));
onMouseLeave: this.handleHideTooltip,
className: className
}, children));
}

@@ -204,0 +202,0 @@ }]);

{
"name": "@ndla/tooltip",
"version": "0.2.11",
"version": "0.2.12",
"description": "Tooltip component from NDLA",

@@ -22,3 +22,3 @@ "license": "GPL-3.0",

"dependencies": {
"@ndla/core": "^0.6.11"
"@ndla/core": "^0.6.12"
},

@@ -35,3 +35,3 @@ "peerDependencies": {

},
"gitHead": "f64e39c93f24d9b0f3255f80b9d8a33d26691aca"
"gitHead": "ffba847067bb684b37789a5fb071b5720d2e49cf"
}
SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc