@ndla/tooltip
Advanced tools
Comparing version 0.2.11 to 0.2.12
@@ -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" | ||
} |
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
New author
Supply chain riskA new npm collaborator published a version of the package for the first time. New collaborators are usually benign additions to a project, but do indicate a change to the security surface area of a package.
Found 1 instance in 1 package
Long strings
Supply chain riskContains long string literals, which may be a sign of obfuscated or packed code.
Found 1 instance in 1 package
12
105439
584
Updated@ndla/core@^0.6.12