@ndla/ui 34.6.5 → 34.6.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/es/AudioPlayer/AudioPlayer.js +19 -19
- package/es/ErrorMessage/ErrorResourceAccessDenied.js +3 -3
- package/es/FactBox/FactBox.js +3 -3
- package/es/Figure/Figure.js +8 -10
- package/es/Filter/FilterButtons.js +20 -23
- package/es/Filter/FilterListPhone.js +7 -7
- package/es/Footer/FooterAuth.js +12 -11
- package/es/Frontpage/FrontpageProgramMenu.js +16 -16
- package/es/LearningPaths/LearningPathMenuModalWrapper.js +3 -3
- package/es/Navigation/NavigationTopicAbout.js +17 -17
- package/es/NoContentBox/NoContentBox.js +3 -3
- package/es/RelatedArticleList/RelatedArticleList.js +3 -3
- package/es/Search/ContentTypeResult.js +5 -4
- package/es/Search/ToggleSearchButton.js +4 -4
- package/es/SearchTypeResult/ActiveFilters.js +9 -10
- package/es/SearchTypeResult/PopupFilter.js +14 -14
- package/es/SearchTypeResult/ResultNavigation.js +9 -9
- package/es/SearchTypeResult/SearchHeader.js +11 -11
- package/es/SearchTypeResult/SearchNotionItem.js +18 -18
- package/es/SearchTypeResult/SearchTypeHeader.js +13 -13
- package/es/SearchTypeResult/SearchViewType.js +9 -8
- package/es/SearchTypeResult/components/SubjectFilters.js +5 -5
- package/es/User/AuthModal.js +9 -9
- package/lib/AudioPlayer/AudioPlayer.js +19 -19
- package/lib/ErrorMessage/ErrorResourceAccessDenied.js +3 -3
- package/lib/FactBox/FactBox.js +3 -3
- package/lib/Figure/Figure.js +8 -10
- package/lib/Filter/FilterButtons.js +20 -23
- package/lib/Filter/FilterListPhone.js +7 -7
- package/lib/Footer/FooterAuth.js +12 -11
- package/lib/Frontpage/FrontpageProgramMenu.js +16 -16
- package/lib/LearningPaths/LearningPathMenuModalWrapper.js +3 -3
- package/lib/Navigation/NavigationTopicAbout.js +17 -17
- package/lib/NoContentBox/NoContentBox.js +3 -3
- package/lib/RelatedArticleList/RelatedArticleList.js +3 -3
- package/lib/Search/ContentTypeResult.js +5 -4
- package/lib/Search/ToggleSearchButton.js +4 -4
- package/lib/SearchTypeResult/ActiveFilters.js +9 -10
- package/lib/SearchTypeResult/PopupFilter.js +14 -14
- package/lib/SearchTypeResult/ResultNavigation.js +9 -9
- package/lib/SearchTypeResult/SearchHeader.js +11 -11
- package/lib/SearchTypeResult/SearchNotionItem.js +18 -18
- package/lib/SearchTypeResult/SearchTypeHeader.js +13 -13
- package/lib/SearchTypeResult/SearchViewType.js +9 -8
- package/lib/SearchTypeResult/components/SubjectFilters.js +5 -5
- package/lib/User/AuthModal.js +9 -9
- package/package.json +2 -2
- package/src/AudioPlayer/AudioPlayer.tsx +9 -9
- package/src/ErrorMessage/ErrorResourceAccessDenied.tsx +3 -3
- package/src/FactBox/FactBox.tsx +3 -3
- package/src/Figure/Figure.tsx +7 -13
- package/src/Filter/FilterButtons.tsx +12 -14
- package/src/Filter/FilterListPhone.tsx +7 -7
- package/src/Footer/FooterAuth.tsx +4 -4
- package/src/Frontpage/FrontpageProgramMenu.tsx +13 -5
- package/src/LearningPaths/LearningPathMenuModalWrapper.tsx +3 -3
- package/src/Navigation/NavigationTopicAbout.tsx +4 -4
- package/src/NoContentBox/NoContentBox.tsx +3 -3
- package/src/RelatedArticleList/RelatedArticleList.tsx +4 -4
- package/src/Search/ContentTypeResult.tsx +5 -4
- package/src/Search/ToggleSearchButton.tsx +2 -2
- package/src/SearchTypeResult/ActiveFilters.tsx +4 -5
- package/src/SearchTypeResult/PopupFilter.tsx +9 -9
- package/src/SearchTypeResult/ResultNavigation.tsx +5 -5
- package/src/SearchTypeResult/SearchHeader.tsx +3 -3
- package/src/SearchTypeResult/SearchNotionItem.tsx +5 -5
- package/src/SearchTypeResult/SearchTypeHeader.tsx +5 -5
- package/src/SearchTypeResult/SearchViewType.tsx +5 -4
- package/src/SearchTypeResult/components/SubjectFilters.tsx +3 -3
- package/src/User/AuthModal.tsx +3 -3
|
@@ -16,7 +16,7 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
|
|
|
16
16
|
|
|
17
17
|
import React, { useEffect, useRef, useState } from 'react';
|
|
18
18
|
import { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';
|
|
19
|
-
import
|
|
19
|
+
import { ButtonV2 } from '@ndla/button';
|
|
20
20
|
import { Cross as CrossIcon } from '@ndla/icons/action';
|
|
21
21
|
import { useTranslation } from 'react-i18next';
|
|
22
22
|
import SafeLink from '@ndla/safelink';
|
|
@@ -31,7 +31,7 @@ var InfoWrapper = /*#__PURE__*/_styled("div", {
|
|
|
31
31
|
label: "InfoWrapper"
|
|
32
32
|
})("border:1px solid ", colors.brand.lighter, ";border-bottom:0;display:flex;", mq.range({
|
|
33
33
|
until: breakpoints.tabletWide
|
|
34
|
-
}), "{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"]} */"));
|
|
34
|
+
}), "{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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
35
35
|
var ImageWrapper = /*#__PURE__*/_styled("div", {
|
|
36
36
|
target: "e1wmpntw12",
|
|
37
37
|
label: "ImageWrapper"
|
|
@@ -39,7 +39,7 @@ var ImageWrapper = /*#__PURE__*/_styled("div", {
|
|
|
39
39
|
from: breakpoints.desktop
|
|
40
40
|
}), "{width:260px;height:260px;}", mq.range({
|
|
41
41
|
until: breakpoints.tabletWide
|
|
42
|
-
}), "{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"]} */"));
|
|
42
|
+
}), "{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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
43
43
|
var TextWrapper = /*#__PURE__*/_styled("div", {
|
|
44
44
|
target: "e1wmpntw11",
|
|
45
45
|
label: "TextWrapper"
|
|
@@ -49,27 +49,27 @@ var TextWrapper = /*#__PURE__*/_styled("div", {
|
|
|
49
49
|
}), " {\n padding: ").concat(spacing.small, " ").concat(spacing.normal, ";\n }\n ").concat(mq.range({
|
|
50
50
|
from: breakpoints.tabletWide
|
|
51
51
|
}), " {\n padding: ").concat(spacing.small, " ").concat(spacing.medium, ";\n }");
|
|
52
|
-
}, ";" + (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"]} */"));
|
|
52
|
+
}, ";" + (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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
53
53
|
var TitleWrapper = /*#__PURE__*/_styled("div", {
|
|
54
54
|
target: "e1wmpntw10",
|
|
55
55
|
label: "TitleWrapper"
|
|
56
56
|
})(mq.range({
|
|
57
57
|
from: breakpoints.tabletWide
|
|
58
|
-
}), "{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"]} */"));
|
|
58
|
+
}), "{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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
59
59
|
var Title = /*#__PURE__*/_styled("h2", {
|
|
60
60
|
target: "e1wmpntw9",
|
|
61
61
|
label: "Title"
|
|
62
62
|
})(fonts.sizes('22px', '30px'), ";margin:0 0 ", function (props) {
|
|
63
63
|
return props.hasDescription && "".concat(spacing.small);
|
|
64
|
-
}, ";" + (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"]} */"));
|
|
64
|
+
}, ";" + (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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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
65
|
var Subtitle = /*#__PURE__*/_styled("h3", {
|
|
66
66
|
target: "e1wmpntw8",
|
|
67
67
|
label: "Subtitle"
|
|
68
|
-
})(fonts.sizes('18px', '28px'), ";margin:0;font-weight:", 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"]} */"));
|
|
68
|
+
})(fonts.sizes('18px', '28px'), ";margin:0;font-weight:", 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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
69
69
|
var StyledDescription = /*#__PURE__*/_styled("div", {
|
|
70
70
|
target: "e1wmpntw7",
|
|
71
71
|
label: "StyledDescription"
|
|
72
|
-
})(fonts.sizes('16px', '30px'), ";font-family:", 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"]} */"));
|
|
72
|
+
})(fonts.sizes('16px', '30px'), ";font-family:", 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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
73
73
|
var LinkToTextVersionWrapper = /*#__PURE__*/_styled("div", {
|
|
74
74
|
target: "e1wmpntw6",
|
|
75
75
|
label: "LinkToTextVersionWrapper"
|
|
@@ -77,7 +77,7 @@ var LinkToTextVersionWrapper = /*#__PURE__*/_styled("div", {
|
|
|
77
77
|
return !props.noMargin && "margin-top: ".concat(spacing.normal, ";\n ");
|
|
78
78
|
}, " ", mq.range({
|
|
79
79
|
until: breakpoints.tabletWide
|
|
80
|
-
}), "{margin:", 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"]} */"));
|
|
80
|
+
}), "{margin:", 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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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
81
|
var TextVersionWrapper = /*#__PURE__*/_styled("div", {
|
|
82
82
|
target: "e1wmpntw5",
|
|
83
83
|
label: "TextVersionWrapper"
|
|
@@ -85,7 +85,7 @@ var TextVersionWrapper = /*#__PURE__*/_styled("div", {
|
|
|
85
85
|
from: breakpoints.tablet
|
|
86
86
|
}), "{padding:", spacing.normal, ";}", mq.range({
|
|
87
87
|
from: breakpoints.tabletWide
|
|
88
|
-
}), "{padding:", spacing.normal, " ", 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"]} */"));
|
|
88
|
+
}), "{padding:", spacing.normal, " ", 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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
89
89
|
var TextVersionHeadingWrapper = /*#__PURE__*/_styled("div", {
|
|
90
90
|
target: "e1wmpntw4",
|
|
91
91
|
label: "TextVersionHeadingWrapper"
|
|
@@ -95,21 +95,21 @@ var TextVersionHeadingWrapper = /*#__PURE__*/_styled("div", {
|
|
|
95
95
|
} : {
|
|
96
96
|
name: "1f1m9i6",
|
|
97
97
|
styles: "display:flex;justify-content:space-between;align-items:flex-start",
|
|
98
|
-
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"]} */",
|
|
98
|
+
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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */",
|
|
99
99
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
100
100
|
});
|
|
101
101
|
var TextVersionHeading = /*#__PURE__*/_styled("h2", {
|
|
102
102
|
target: "e1wmpntw3",
|
|
103
103
|
label: "TextVersionHeading"
|
|
104
|
-
})("font-weight:", fonts.weight.semibold, ";margin:", spacing.small, " 0 ", 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"]} */"));
|
|
105
|
-
var LinkButton = /*#__PURE__*/_styled(
|
|
104
|
+
})("font-weight:", fonts.weight.semibold, ";margin:", spacing.small, " 0 ", 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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
105
|
+
var LinkButton = /*#__PURE__*/_styled(ButtonV2, {
|
|
106
106
|
target: "e1wmpntw2",
|
|
107
107
|
label: "LinkButton"
|
|
108
|
-
})("box-shadow:none;padding-left:0;padding-right:4px;min-height:", spacing.medium, ";", fonts.sizes('16px', '25px'), ";flex:0 0 auto;&:hover,&:focus{box-shadow:", 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"]} */"));
|
|
108
|
+
})("box-shadow:none;padding-left:0;padding-right:4px;min-height:", spacing.medium, ";", fonts.sizes('16px', '25px'), ";flex:0 0 auto;&:hover,&:focus{box-shadow:", colors.link, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["AudioPlayer.tsx"],"names":[],"mappings":"AA+ImC","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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
109
109
|
var CloseText = /*#__PURE__*/_styled("span", {
|
|
110
110
|
target: "e1wmpntw1",
|
|
111
111
|
label: "CloseText"
|
|
112
|
-
})("display:inline-block;margin-left:", 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"]} */"));
|
|
112
|
+
})("display:inline-block;margin-left:", 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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */"));
|
|
113
113
|
var TextVersionText = /*#__PURE__*/_styled("div", {
|
|
114
114
|
target: "e1wmpntw0",
|
|
115
115
|
label: "TextVersionText"
|
|
@@ -119,7 +119,7 @@ var TextVersionText = /*#__PURE__*/_styled("div", {
|
|
|
119
119
|
} : {
|
|
120
120
|
name: "j8dnu6",
|
|
121
121
|
styles: "max-width:670px",
|
|
122
|
-
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"]} */",
|
|
122
|
+
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 { ButtonV2 } 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(ButtonV2)`\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      <ButtonV2 size=\"normal\" shape=\"pill\" onClick={toggleTextVersion} data-audio-text-button-id={staticRenderId}>\n        {t('audio.textVersion.heading')}\n      </ButtonV2>\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\n              variant=\"link\"\n              size=\"normal\"\n              onClick={toggleTextVersion}\n              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"]} */",
|
|
123
123
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
124
124
|
});
|
|
125
125
|
export var truncateDescription = function truncateDescription(el, readMoreLabel) {
|
|
@@ -168,9 +168,9 @@ var AudioPlayer = function AudioPlayer(_ref) {
|
|
|
168
168
|
var noMargin = _ref2.noMargin;
|
|
169
169
|
return _jsx(LinkToTextVersionWrapper, {
|
|
170
170
|
noMargin: noMargin,
|
|
171
|
-
children: _jsx(
|
|
171
|
+
children: _jsx(ButtonV2, {
|
|
172
172
|
size: "normal",
|
|
173
|
-
|
|
173
|
+
shape: "pill",
|
|
174
174
|
onClick: toggleTextVersion,
|
|
175
175
|
"data-audio-text-button-id": staticRenderId,
|
|
176
176
|
children: t('audio.textVersion.heading')
|
|
@@ -224,7 +224,7 @@ var AudioPlayer = function AudioPlayer(_ref) {
|
|
|
224
224
|
children: [_jsx(TextVersionHeading, {
|
|
225
225
|
children: t('audio.textVersion.heading')
|
|
226
226
|
}), _jsxs(LinkButton, {
|
|
227
|
-
|
|
227
|
+
variant: "link",
|
|
228
228
|
size: "normal",
|
|
229
229
|
onClick: toggleTextVersion,
|
|
230
230
|
"data-audio-text-button-id": staticRenderId,
|