@ndla/ui 4.0.0 → 4.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (116) hide show
  1. package/es/Article/ArticleAuthorContent.js +1 -2
  2. package/es/Article/ArticleByline.js +5 -6
  3. package/es/Article/ArticleHeaderWrapper.js +5 -6
  4. package/es/Article/ArticleSideBar.js +5 -6
  5. package/es/AudioPlayer/AudioPlayer.js +15 -16
  6. package/es/CompetenceGoalTab/CompetenceGoalTab.js +6 -7
  7. package/es/ErrorMessage/ErrorResourceAccessDenied.js +2 -3
  8. package/es/FactBox/FactBox.js +1 -2
  9. package/es/Figure/Figure.js +1 -2
  10. package/es/Filter/FilterButtons.js +9 -10
  11. package/es/Footer/FooterAuth.js +8 -9
  12. package/es/Frontpage/FrontpageProgramMenu.js +10 -11
  13. package/es/LanguageSelector/LanguageSelector.js +8 -9
  14. package/es/LearningPaths/LearningPathMenuModalWrapper.js +2 -3
  15. package/es/Masthead/MastheadAuthModal.js +1 -2
  16. package/es/Masthead/MastheadSearchModal.js +4 -5
  17. package/es/NDLAFilm/AboutNdlaFilm.js +2 -3
  18. package/es/Navigation/NavigationBox.js +23 -22
  19. package/es/Navigation/NavigationTopicAbout.js +15 -16
  20. package/es/NoContentBox/NoContentBox.js +1 -2
  21. package/es/Notion/Notion.js +5 -6
  22. package/es/Programme/ProgrammeSubjects.js +1 -2
  23. package/es/RelatedArticleList/RelatedArticleList.js +1 -2
  24. package/es/Search/ContentTypeResult.js +2 -3
  25. package/es/SearchTypeResult/ActiveFilterContent.js +2 -3
  26. package/es/SearchTypeResult/PopupFilter.js +9 -10
  27. package/es/SearchTypeResult/ResultNavigation.js +3 -4
  28. package/es/SearchTypeResult/SearchHeader.js +10 -11
  29. package/es/SearchTypeResult/SearchItem.js +18 -19
  30. package/es/SearchTypeResult/SearchNotionItem.js +13 -14
  31. package/es/SearchTypeResult/SearchTypeHeader.js +6 -7
  32. package/es/Topic/Topic.js +23 -28
  33. package/es/User/AuthModal.js +9 -10
  34. package/es/locale/messages-en.js +3 -0
  35. package/es/locale/messages-nb.js +3 -0
  36. package/es/locale/messages-nn.js +3 -0
  37. package/lib/Article/ArticleByline.js +4 -4
  38. package/lib/Article/ArticleHeaderWrapper.js +4 -4
  39. package/lib/Article/ArticleSideBar.js +4 -4
  40. package/lib/AudioPlayer/AudioPlayer.js +14 -14
  41. package/lib/CompetenceGoalTab/CompetenceGoalTab.js +5 -5
  42. package/lib/ErrorMessage/ErrorResourceAccessDenied.js +1 -2
  43. package/lib/Filter/FilterButtons.js +9 -9
  44. package/lib/Footer/FooterAuth.js +7 -7
  45. package/lib/Frontpage/FrontpageMultidisciplinarySubject.d.ts +5 -5
  46. package/lib/Frontpage/FrontpageProgramMenu.js +10 -10
  47. package/lib/Frontpage/illustrations/FrontpageIllustrations.d.ts +2 -2
  48. package/lib/LanguageSelector/LanguageSelector.js +7 -7
  49. package/lib/LearningPaths/LearningPathMenuModalWrapper.js +1 -2
  50. package/lib/Masthead/MastheadAuthModal.js +1 -1
  51. package/lib/Masthead/MastheadSearchModal.js +3 -4
  52. package/lib/MessageBox/MessageBox.d.ts +1 -1
  53. package/lib/MultidisciplinarySubject/Illustrations.d.ts +7 -7
  54. package/lib/NDLAFilm/AboutNdlaFilm.js +1 -2
  55. package/lib/Navigation/NavigationBox.js +22 -20
  56. package/lib/Navigation/NavigationTopicAbout.js +14 -14
  57. package/lib/NoContentBox/NoContentBox.js +0 -1
  58. package/lib/Notion/Notion.js +4 -4
  59. package/lib/Programme/ProgrammeSubjects.js +1 -2
  60. package/lib/Search/ContentTypeResult.js +1 -1
  61. package/lib/Search/ContentTypeResultStyles.d.ts +3 -3
  62. package/lib/SearchTypeResult/ActiveFilterContent.d.ts +1 -1
  63. package/lib/SearchTypeResult/ActiveFilterContent.js +1 -2
  64. package/lib/SearchTypeResult/PopupFilter.js +8 -8
  65. package/lib/SearchTypeResult/ResultNavigation.js +2 -3
  66. package/lib/SearchTypeResult/SearchHeader.d.ts +1 -1
  67. package/lib/SearchTypeResult/SearchHeader.js +9 -9
  68. package/lib/SearchTypeResult/SearchItem.js +17 -17
  69. package/lib/SearchTypeResult/SearchNotionItem.js +12 -12
  70. package/lib/SearchTypeResult/SearchTypeHeader.d.ts +1 -1
  71. package/lib/SearchTypeResult/SearchTypeHeader.js +6 -6
  72. package/lib/Topic/Topic.js +22 -26
  73. package/lib/User/AuthModal.js +8 -8
  74. package/lib/locale/messages-en.d.ts +3 -0
  75. package/lib/locale/messages-en.js +3 -0
  76. package/lib/locale/messages-nb.d.ts +3 -0
  77. package/lib/locale/messages-nb.js +3 -0
  78. package/lib/locale/messages-nn.d.ts +3 -0
  79. package/lib/locale/messages-nn.js +3 -0
  80. package/package.json +6 -6
  81. package/src/Article/ArticleAuthorContent.tsx +0 -1
  82. package/src/Article/ArticleByline.tsx +0 -1
  83. package/src/Article/ArticleHeaderWrapper.tsx +0 -1
  84. package/src/Article/ArticleSideBar.tsx +0 -1
  85. package/src/AudioPlayer/AudioPlayer.tsx +0 -1
  86. package/src/CompetenceGoalTab/CompetenceGoalTab.tsx +0 -1
  87. package/src/ErrorMessage/ErrorResourceAccessDenied.tsx +0 -1
  88. package/src/FactBox/FactBox.tsx +0 -1
  89. package/src/Figure/Figure.tsx +0 -1
  90. package/src/Filter/FilterButtons.tsx +0 -1
  91. package/src/Footer/FooterAuth.tsx +0 -1
  92. package/src/Frontpage/FrontpageProgramMenu.tsx +0 -1
  93. package/src/LanguageSelector/LanguageSelector.tsx +0 -1
  94. package/src/LearningPaths/LearningPathMenuModalWrapper.tsx +0 -1
  95. package/src/Masthead/MastheadAuthModal.tsx +1 -2
  96. package/src/Masthead/MastheadSearchModal.tsx +0 -1
  97. package/src/NDLAFilm/AboutNdlaFilm.tsx +0 -1
  98. package/src/Navigation/NavigationBox.tsx +1 -2
  99. package/src/Navigation/NavigationTopicAbout.tsx +0 -1
  100. package/src/NoContentBox/NoContentBox.tsx +0 -1
  101. package/src/Notion/Notion.tsx +0 -2
  102. package/src/Programme/ProgrammeSubjects.tsx +0 -1
  103. package/src/RelatedArticleList/RelatedArticleList.tsx +0 -1
  104. package/src/Search/ContentTypeResult.tsx +0 -1
  105. package/src/SearchTypeResult/ActiveFilterContent.tsx +0 -1
  106. package/src/SearchTypeResult/PopupFilter.tsx +0 -1
  107. package/src/SearchTypeResult/ResultNavigation.tsx +0 -1
  108. package/src/SearchTypeResult/SearchHeader.tsx +0 -1
  109. package/src/SearchTypeResult/SearchItem.tsx +0 -1
  110. package/src/SearchTypeResult/SearchNotionItem.tsx +0 -1
  111. package/src/SearchTypeResult/SearchTypeHeader.tsx +0 -1
  112. package/src/Topic/Topic.tsx +0 -5
  113. package/src/User/AuthModal.tsx +0 -1
  114. package/src/locale/messages-en.ts +3 -0
  115. package/src/locale/messages-nb.ts +3 -0
  116. package/src/locale/messages-nn.ts +3 -0
@@ -54,7 +54,7 @@ var InfoWrapper = (0, _styledBase["default"])("div", {
54
54
  label: "InfoWrapper"
55
55
  })("border:1px solid ", _core.colors.brand.lighter, ";border-bottom:0;display:flex;", _core.mq.range({
56
56
  until: _core.breakpoints.tabletWide
57
- }), "{display:block;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAoB8B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
57
+ }), "{display:block;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAmB8B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
58
58
  var ImageWrapper = (0, _styledBase["default"])("div", {
59
59
  target: "e1wmpntw1",
60
60
  label: "ImageWrapper"
@@ -62,7 +62,7 @@ var ImageWrapper = (0, _styledBase["default"])("div", {
62
62
  from: _core.breakpoints.desktop
63
63
  }), "{width:260px;height:260px;}", _core.mq.range({
64
64
  until: _core.breakpoints.tabletWide
65
- }), "{max-height:400px;max-width:100%;width:100%;height:auto;img{object-fit:scale-down;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA6B+B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
65
+ }), "{max-height:400px;max-width:100%;width:100%;height:auto;img{object-fit:scale-down;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA4B+B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
66
66
  var TextWrapper = (0, _styledBase["default"])("div", {
67
67
  target: "e1wmpntw2",
68
68
  label: "TextWrapper"
@@ -72,27 +72,27 @@ var TextWrapper = (0, _styledBase["default"])("div", {
72
72
  }), " {\n padding: ").concat(_core.spacing.small, " ").concat(_core.spacing.normal, ";\n }\n ").concat(_core.mq.range({
73
73
  from: _core.breakpoints.tabletWide
74
74
  }), " {\n padding: ").concat(_core.spacing.small, " ").concat(_core.spacing.medium, ";\n }");
75
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA6DgD","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */");
75
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA4DgD","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */");
76
76
  var TitleWrapper = (0, _styledBase["default"])("div", {
77
77
  target: "e1wmpntw3",
78
78
  label: "TitleWrapper"
79
79
  })(_core.mq.range({
80
80
  from: _core.breakpoints.tabletWide
81
- }), "{display:flex;justify-content:space-between;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA2E+B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
81
+ }), "{display:flex;justify-content:space-between;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA0E+B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
82
82
  var Title = (0, _styledBase["default"])("h2", {
83
83
  target: "e1wmpntw4",
84
84
  label: "Title"
85
85
  })(_core.fonts.sizes('22px', '30px'), ";margin:0 0 ", function (props) {
86
86
  return props.hasDescription && "".concat(_core.spacing.small);
87
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAsFmC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
87
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAqFmC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
88
88
  var Subtitle = (0, _styledBase["default"])("h3", {
89
89
  target: "e1wmpntw5",
90
90
  label: "Subtitle"
91
- })(_core.fonts.sizes('18px', '28px'), ";margin:0;font-weight:", _core.fonts.weight.semibold, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA2F0B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
91
+ })(_core.fonts.sizes('18px', '28px'), ";margin:0;font-weight:", _core.fonts.weight.semibold, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA0F0B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
92
92
  var StyledDescription = (0, _styledBase["default"])("div", {
93
93
  target: "e1wmpntw6",
94
94
  label: "StyledDescription"
95
- })(_core.fonts.sizes('16px', '30px'), ";font-family:", _core.fonts.sans, ";margin:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAiGoC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
95
+ })(_core.fonts.sizes('16px', '30px'), ";font-family:", _core.fonts.sans, ";margin:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAgGoC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
96
96
  var LinkToTextVersionWrapper = (0, _styledBase["default"])("div", {
97
97
  target: "e1wmpntw7",
98
98
  label: "LinkToTextVersionWrapper"
@@ -100,7 +100,7 @@ var LinkToTextVersionWrapper = (0, _styledBase["default"])("div", {
100
100
  return !props.noMargin && "margin-top: ".concat(_core.spacing.normal, ";\n ");
101
101
  }, " ", _core.mq.range({
102
102
  until: _core.breakpoints.tabletWide
103
- }), "{margin:", _core.spacing.small, " 0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA0G0E","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
103
+ }), "{margin:", _core.spacing.small, " 0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAyG0E","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
104
104
  var TextVersionWrapper = (0, _styledBase["default"])("div", {
105
105
  target: "e1wmpntw8",
106
106
  label: "TextVersionWrapper"
@@ -108,7 +108,7 @@ var TextVersionWrapper = (0, _styledBase["default"])("div", {
108
108
  from: _core.breakpoints.tablet
109
109
  }), "{padding:", _core.spacing.normal, ";}", _core.mq.range({
110
110
  from: _core.breakpoints.tabletWide
111
- }), "{padding:", _core.spacing.normal, " ", _core.spacing.medium, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAoHqC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
111
+ }), "{padding:", _core.spacing.normal, " ", _core.spacing.medium, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAmHqC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
112
112
  var TextVersionHeadingWrapper = (0, _styledBase["default"])("div", {
113
113
  target: "e1wmpntw9",
114
114
  label: "TextVersionHeadingWrapper"
@@ -118,21 +118,21 @@ var TextVersionHeadingWrapper = (0, _styledBase["default"])("div", {
118
118
  } : {
119
119
  name: "1qdyvok",
120
120
  styles: "display:flex;justify-content:space-between;align-items:flex-start;",
121
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAqI4C","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */",
121
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAoI4C","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */",
122
122
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
123
123
  });
124
124
  var TextVersionHeading = (0, _styledBase["default"])("h2", {
125
125
  target: "e1wmpntw10",
126
126
  label: "TextVersionHeading"
127
- })("font-weight:", _core.fonts.weight.semibold, ";margin:", _core.spacing.small, " 0 ", _core.spacing.normal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA2IoC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
127
+ })("font-weight:", _core.fonts.weight.semibold, ";margin:", _core.spacing.small, " 0 ", _core.spacing.normal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA0IoC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
128
128
  var LinkButton = ( /*#__PURE__*/0, _styledBase["default"])(_button["default"], {
129
129
  target: "e1wmpntw11",
130
130
  label: "LinkButton"
131
- })("box-shadow:none;padding-left:0;padding-right:4px;min-height:", _core.spacing.medium, ";", _core.fonts.sizes('16px', '25px'), ";flex:0 0 auto;&:hover,&:focus{box-shadow:", _core.colors.link, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAgJiC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
131
+ })("box-shadow:none;padding-left:0;padding-right:4px;min-height:", _core.spacing.medium, ";", _core.fonts.sizes('16px', '25px'), ";flex:0 0 auto;&:hover,&:focus{box-shadow:", _core.colors.link, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA+IiC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
132
132
  var CloseText = (0, _styledBase["default"])("span", {
133
133
  target: "e1wmpntw12",
134
134
  label: "CloseText"
135
- })("display:inline-block;margin-left:", _core.spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA6J6B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
135
+ })("display:inline-block;margin-left:", _core.spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA4J6B","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */"));
136
136
  var TextVersionText = (0, _styledBase["default"])("div", {
137
137
  target: "e1wmpntw13",
138
138
  label: "TextVersionText"
@@ -142,7 +142,7 @@ var TextVersionText = (0, _styledBase["default"])("div", {
142
142
  } : {
143
143
  name: "1qyyf3g",
144
144
  styles: "max-width:670px;",
145
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAkKkC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n// @ts-ignore\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */",
145
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AAiKkC","file":"AudioPlayer.tsx","sourcesContent":["/**\n * Copyright (c) 2017-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, { ReactNode, useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport Button from '@ndla/button';\nimport { Cross as CrossIcon } from '@ndla/icons/action';\nimport { useTranslation } from 'react-i18next';\nimport SafeLink from '@ndla/safelink';\nimport shave from 'shave';\nimport Controls from './Controls';\nimport SpeechControl from './SpeechControl';\n\nconst InfoWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-bottom: 0;\n  display: flex;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: block;\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  display: flex;\n  align-items: center;\n  flex: 1 0 auto;\n\n  width: 200px;\n  height: 200px;\n\n  img {\n    width: 100%;\n    height: 100%;\n    object-fit: cover;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    width: 260px;\n    height: 260px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    max-height: 400px;\n    max-width: 100%;\n    width: 100%;\n    height: auto;\n    img {\n      object-fit: scale-down;\n    }\n  }\n`;\n\ntype TextWrapperProps = {\n  hasImage?: boolean;\n};\n\nconst TextWrapper = styled.div<TextWrapperProps>`\n  padding: ${spacing.small};\n  width: 100%;\n\n  ${(props) =>\n    props.hasImage &&\n    `${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.medium};\n  }`}\n`;\n\nconst TitleWrapper = styled.div`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    display: flex;\n    justify-content: space-between;\n  }\n`;\n\ntype TitleProps = {\n  hasDescription?: boolean;\n};\n\nconst Title = styled.h2<TitleProps>`\n  ${fonts.sizes('22px', '30px')};\n  margin: 0 0 ${(props) => props.hasDescription && `${spacing.small}`};\n`;\n\nconst Subtitle = styled.h3`\n  ${fonts.sizes('18px', '28px')};\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst StyledDescription = styled.div`\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  margin: 0;\n`;\n\ntype LinkToTextVersionWrapperProps = {\n  noMargin?: boolean;\n};\nconst LinkToTextVersionWrapper = styled.div<LinkToTextVersionWrapperProps>`\n  ${(props) =>\n    !props.noMargin &&\n    `margin-top: ${spacing.normal};\n  `}\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    margin: ${spacing.small} 0;\n  }\n`;\n\nconst TextVersionWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  border-top: 0;\n  ${fonts.sizes('16px', '30px')};\n  font-family: ${fonts.sans};\n  &.audio-player-text-version-hidden {\n    display: none;\n  }\n  padding: ${spacing.normal} ${spacing.small} ${spacing.small};\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding: ${spacing.normal};\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.normal} ${spacing.medium};\n  }\n`;\n\nconst TextVersionHeadingWrapper = styled.div`\n  display: flex;\n  justify-content: space-between;\n  align-items: flex-start;\n`;\n\nconst TextVersionHeading = styled.h2`\n  font-weight: ${fonts.weight.semibold};\n  margin: ${spacing.small} 0 ${spacing.normal};\n`;\n\nconst LinkButton = styled(Button)`\n  box-shadow: none;\n  padding-left: 0;\n  padding-right: 4px;\n  min-height: ${spacing.medium};\n  ${fonts.sizes('16px', '25px')};\n  flex: 0 0 auto;\n  &:hover,\n  &:focus {\n    box-shadow: ${colors.link};\n  }\n`;\n\nconst CloseText = styled.span`\n  display: inline-block;\n  margin-left: ${spacing.xsmall};\n`;\n\nconst TextVersionText = styled.div`\n  max-width: 670px;\n`;\n\nexport const truncateDescription = (el: HTMLElement, readMoreLabel: string | null) => {\n  shave(el, 90, {\n    character: `... <a href=\"#\" onclick=\"(function(e){e.preventDefault(); const parentNode = e.target.parentNode; parentNode.nextSibling.style.display = 'inline'; parentNode.remove();return false;})(arguments[0]);return false;\">${readMoreLabel}</a>`,\n  });\n};\n\ntype Props = {\n  src: string;\n  title: string;\n  subtitle?: {\n    title: string;\n    url?: string;\n  };\n  speech?: boolean;\n  description?: ReactNode;\n  textVersion?: ReactNode;\n  img?: {\n    url: string;\n    alt: string;\n  };\n  staticRenderId?: string;\n};\n\nconst AudioPlayer = ({ src, title, subtitle, speech, description, img, textVersion, staticRenderId }: Props) => {\n  const { t } = useTranslation();\n  const [showTextVersion, setShowTextVersion] = useState(false);\n\n  const descriptionRef = useRef<HTMLDivElement>(null);\n  const readMoreDescriptionLabel = t('audio.readMoreDescriptionLabel');\n\n  useEffect(() => {\n    if (descriptionRef?.current) {\n      truncateDescription(descriptionRef.current, readMoreDescriptionLabel);\n    }\n  }, [readMoreDescriptionLabel]);\n\n  if (speech) {\n    return (\n      <div data-audio-player={1} data-speech={1} data-src={src} data-title={title}>\n        <SpeechControl src={src} title={title} />\n      </div>\n    );\n  }\n\n  const toggleTextVersion = () => {\n    setShowTextVersion(!showTextVersion);\n  };\n\n  type TextVersionComponentProps = {\n    noMargin?: boolean;\n  };\n  const TextVersionComponent = ({ noMargin }: TextVersionComponentProps) => (\n    <LinkToTextVersionWrapper noMargin={noMargin}>\n      <Button\n        size=\"normal\"\n        borderShape=\"rounded\"\n        onClick={toggleTextVersion}\n        data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </Button>\n    </LinkToTextVersionWrapper>\n  );\n\n  return (\n    <>\n      <InfoWrapper>\n        {img && (\n          <ImageWrapper>\n            <img src={img.url} alt={img.alt} />\n          </ImageWrapper>\n        )}\n        <TextWrapper hasImage={!!img}>\n          <TitleWrapper>\n            <div>\n              {subtitle && (\n                <Subtitle>\n                  {subtitle.url ? <SafeLink to={subtitle.url}>{subtitle.title}</SafeLink> : subtitle.title}\n                </Subtitle>\n              )}\n              <Title hasDescription={!!description}>{title}</Title>\n            </div>\n            {textVersion && !img && <TextVersionComponent noMargin />}\n          </TitleWrapper>\n          {description && (\n            <StyledDescription>\n              <div\n                ref={descriptionRef}\n                data-audio-player-description={1}\n                data-read-more-text={t('audio.readMoreDescriptionLabel')}>\n                {description}\n              </div>\n            </StyledDescription>\n          )}\n          {textVersion && img && <TextVersionComponent />}\n        </TextWrapper>\n      </InfoWrapper>\n      <div data-audio-player={1} data-src={src} data-title={title}>\n        <Controls src={src} title={title} />\n      </div>\n      {textVersion && (showTextVersion || staticRenderId) && (\n        <TextVersionWrapper id={staticRenderId} hidden={!!staticRenderId}>\n          <TextVersionHeadingWrapper>\n            <TextVersionHeading>{t('audio.textVersion.heading')}</TextVersionHeading>\n            <LinkButton link size=\"normal\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n              <CrossIcon style={{ width: '20px', height: '20px' }} />\n              <CloseText>{t('audio.textVersion.close')}</CloseText>\n            </LinkButton>\n          </TextVersionHeadingWrapper>\n          <TextVersionText>{textVersion}</TextVersionText>\n        </TextVersionWrapper>\n      )}\n    </>\n  );\n};\n\nexport default AudioPlayer;\n"]} */",
146
146
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
147
147
  });
148
148