@ndla/ui 46.1.0 → 47.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (55) hide show
  1. package/es/Article/Article.js +3 -17
  2. package/es/Embed/AudioEmbed.js +6 -14
  3. package/es/Embed/BrightcoveEmbed.js +13 -5
  4. package/es/Embed/ConceptEmbed.js +16 -14
  5. package/es/Embed/ImageEmbed.js +11 -4
  6. package/es/Embed/conceptComponents.js +11 -12
  7. package/es/Footer/FooterLinks.js +6 -6
  8. package/es/LicenseByline/EmbedByline.js +7 -15
  9. package/es/Messages/MessageBanner.js +4 -13
  10. package/es/Messages/MessageBox.js +10 -20
  11. package/es/Navigation/NavigationTopicAbout.js +15 -17
  12. package/es/Notion/Notion.js +8 -7
  13. package/es/SearchTypeResult/SearchFieldHeader.js +4 -4
  14. package/es/Topic/Topic.js +18 -20
  15. package/lib/Article/Article.d.ts +2 -4
  16. package/lib/Article/Article.js +3 -17
  17. package/lib/Embed/AudioEmbed.js +6 -15
  18. package/lib/Embed/BrightcoveEmbed.js +12 -4
  19. package/lib/Embed/ConceptEmbed.js +15 -13
  20. package/lib/Embed/ImageEmbed.js +10 -3
  21. package/lib/Embed/conceptComponents.d.ts +1 -1
  22. package/lib/Embed/conceptComponents.js +11 -12
  23. package/lib/Footer/FooterLinks.js +6 -6
  24. package/lib/LicenseByline/EmbedByline.d.ts +2 -2
  25. package/lib/LicenseByline/EmbedByline.js +8 -16
  26. package/lib/Messages/MessageBanner.d.ts +2 -1
  27. package/lib/Messages/MessageBanner.js +5 -13
  28. package/lib/Messages/MessageBox.d.ts +2 -1
  29. package/lib/Messages/MessageBox.js +11 -19
  30. package/lib/Navigation/NavigationTopicAbout.d.ts +2 -3
  31. package/lib/Navigation/NavigationTopicAbout.js +15 -17
  32. package/lib/Notion/Notion.js +7 -6
  33. package/lib/SearchTypeResult/SearchFieldHeader.js +4 -4
  34. package/lib/TagSelector/ariaMessages.d.ts +1 -1
  35. package/lib/Topic/Topic.d.ts +2 -3
  36. package/lib/Topic/Topic.js +18 -20
  37. package/lib/types.d.ts +1 -1
  38. package/package.json +19 -20
  39. package/src/Article/Article.tsx +2 -21
  40. package/src/Embed/AudioEmbed.tsx +5 -11
  41. package/src/Embed/BrightcoveEmbed.tsx +10 -7
  42. package/src/Embed/ConceptEmbed.stories.tsx +1 -1
  43. package/src/Embed/ConceptEmbed.tsx +9 -4
  44. package/src/Embed/ImageEmbed.tsx +12 -2
  45. package/src/Embed/conceptComponents.tsx +2 -3
  46. package/src/Footer/FooterLinks.tsx +1 -1
  47. package/src/LicenseByline/EmbedByline.tsx +4 -11
  48. package/src/Messages/MessageBanner.tsx +3 -8
  49. package/src/Messages/MessageBox.tsx +3 -8
  50. package/src/Navigation/NavigationTopicAbout.tsx +2 -5
  51. package/src/Notion/Notion.tsx +2 -2
  52. package/src/SearchTypeResult/SearchFieldHeader.tsx +4 -4
  53. package/src/TagSelector/ariaMessages.ts +1 -1
  54. package/src/Topic/Topic.tsx +2 -5
  55. package/src/types.ts +1 -1
@@ -7,7 +7,6 @@ exports.default = void 0;
7
7
  var _base = _interopRequireDefault(require("@emotion/styled/base"));
8
8
  var _react = require("react");
9
9
  var _core = require("@ndla/core");
10
- var _htmlReactParser = _interopRequireDefault(require("html-react-parser"));
11
10
  var _common = require("@ndla/icons/common");
12
11
  var _modal = require("@ndla/modal");
13
12
  var _button = require("@ndla/button");
@@ -32,15 +31,15 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
32
31
  var Wrapper = /*#__PURE__*/(0, _base.default)("div", {
33
32
  target: "e111uita13",
34
33
  label: "Wrapper"
35
- })("display:flex;flex-direction:column;gap:", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAgC0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
34
+ })("display:flex;flex-direction:column;gap:", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+B0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
36
35
  var frameStyle = /*#__PURE__*/(0, _react2.css)(_core.mq.range({
37
36
  from: _core.breakpoints.tabletWide
38
37
  }), "{padding:40px 40px;border:2px solid ", _core.colors.brand.neutral7, ";}", _core.mq.range({
39
38
  from: _core.breakpoints.desktop
40
39
  }), "{padding:40px 80px;}", _core.mq.range({
41
40
  from: '1180px'
42
- }), "{padding:60px 160px;};label:frameStyle;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAsCsB","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
43
- var _invertedStyle = /*#__PURE__*/(0, _react2.css)("color:", _core.colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmD0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
41
+ }), "{padding:60px 160px;};label:frameStyle;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAqCsB","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
42
+ var _invertedStyle = /*#__PURE__*/(0, _react2.css)("color:", _core.colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkD0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
44
43
  var TopicHeaderVisualElementWrapper = /*#__PURE__*/(0, _base.default)("div", {
45
44
  target: "e111uita12",
46
45
  label: "TopicHeaderVisualElementWrapper"
@@ -48,7 +47,7 @@ var TopicHeaderVisualElementWrapper = /*#__PURE__*/(0, _base.default)("div", {
48
47
  from: _core.breakpoints.mobileWide
49
48
  }), "{width:150px;height:150px;}", _core.mq.range({
50
49
  from: _core.breakpoints.tabletWide
51
- }), "{width:200px;height:200px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAuDkD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
50
+ }), "{width:200px;height:200px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAsDkD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
52
51
  var ShowVisualElementWrapper = /*#__PURE__*/(0, _base.default)("div", {
53
52
  target: "e111uita11",
54
53
  label: "ShowVisualElementWrapper"
@@ -58,48 +57,48 @@ var ShowVisualElementWrapper = /*#__PURE__*/(0, _base.default)("div", {
58
57
  } : {
59
58
  name: "dleg4l",
60
59
  styles: "border-radius:50%;width:100%;height:100%;overflow:hidden;aspect-ratio:1;mask-image:radial-gradient(white, black);-webkit-mask-image:-webkit-radial-gradient(white, black)",
61
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAqE2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
60
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoE2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
62
61
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
63
62
  });
64
63
  var VisualElementButton = /*#__PURE__*/(0, _base.default)(_button.ButtonV2, {
65
64
  target: "e111uita10",
66
65
  label: "VisualElementButton"
67
- })("color:", _core.colors.brand.secondary, ";width:100%;height:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+E4C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
66
+ })("color:", _core.colors.brand.secondary, ";width:100%;height:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA8E4C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
68
67
  var TopicHeaderImage = /*#__PURE__*/(0, _base.default)("img", {
69
68
  target: "e111uita9",
70
69
  label: "TopicHeaderImage"
71
- })("border-radius:50%;aspect-ratio:1;width:100%;height:100%;object-fit:cover;transition:transform ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{transform:scale(1.1);opacity:1.2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAqFmC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
70
+ })("border-radius:50%;aspect-ratio:1;width:100%;height:100%;object-fit:cover;transition:transform ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{transform:scale(1.1);opacity:1.2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoFmC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
72
71
  var ExpandVisualElementButton = /*#__PURE__*/(0, _base.default)("span", {
73
72
  target: "e111uita8",
74
73
  label: "ExpandVisualElementButton"
75
74
  })("position:absolute;right:-10px;bottom:-4px;transition:all ", _core.animations.durations.fast, ";svg{width:24px;height:24px;}", VisualElementButton, ":hover &{right:10px;}", _core.mq.range({
76
75
  from: _core.breakpoints.mobileWide
77
- }), "{right:0;bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkG6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
76
+ }), "{right:0;bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiG6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
78
77
  var TopicHeaderOverlay = /*#__PURE__*/(0, _base.default)("div", {
79
78
  target: "e111uita7",
80
79
  label: "TopicHeaderOverlay"
81
- })("background:black;opacity:0.05;position:absolute;top:0;left:0;bottom:0;right:0;border-radius:50%;transition:opacity ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{opacity:0.1;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoHqC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
80
+ })("background:black;opacity:0.05;position:absolute;top:0;left:0;bottom:0;right:0;border-radius:50%;transition:opacity ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{opacity:0.1;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmHqC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
82
81
  var TopicIntroductionWrapper = /*#__PURE__*/(0, _base.default)("div", {
83
82
  target: "e111uita6",
84
83
  label: "TopicIntroductionWrapper"
85
- })("display:flex;gap:", _core.spacing.xsmall, ";justify-content:space-between;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmI2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
84
+ })("display:flex;gap:", _core.spacing.xsmall, ";justify-content:space-between;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkI2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
86
85
  var HeadingWrapper = /*#__PURE__*/(0, _base.default)("hgroup", {
87
86
  target: "e111uita5",
88
87
  label: "HeadingWrapper"
89
- })("display:flex;flex-wrap:wrap;align-items:center;gap:", _core.spacing.small, ";h1{margin:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAyIoC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
88
+ })("display:flex;flex-wrap:wrap;align-items:center;gap:", _core.spacing.small, ";h1{margin:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwIoC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
90
89
  var TopicIntroduction = /*#__PURE__*/(0, _base.default)("div", {
91
90
  target: "e111uita4",
92
91
  label: "TopicIntroduction"
93
92
  })("font-weight:", _core.fonts.weight.light, ";max-width:612px;", _core.mq.range({
94
93
  from: _core.breakpoints.tablet
95
- }), "{", _core.fonts.sizes('22px', '32px'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmJoC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
94
+ }), "{", _core.fonts.sizes('22px', '32px'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkJoC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
96
95
  var _ref = process.env.NODE_ENV === "production" ? {
97
96
  name: "1pu3m2l-StyledButtonWrapper",
98
97
  styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;"
99
98
  } : {
100
99
  name: "1pu3m2l-StyledButtonWrapper",
101
100
  styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;",
102
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiKO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
101
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAgKO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
103
102
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
104
103
  };
105
104
  var StyledButtonWrapper = /*#__PURE__*/(0, _base.default)("div", {
@@ -107,7 +106,7 @@ var StyledButtonWrapper = /*#__PURE__*/(0, _base.default)("div", {
107
106
  label: "StyledButtonWrapper"
108
107
  })("margin-top:", _core.spacing.small, ";padding:", _core.spacing.xsmall, " 0 ", _core.spacing.xsmall, " ", _core.spacing.medium, ";border-left:6px solid ", _core.colors.brand.light, ";", function (props) {
109
108
  return props.invertedStyle && _ref;
110
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA2JqD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
109
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA0JqD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
111
110
  var AdditionalIcon = /*#__PURE__*/(0, _base.default)("span", {
112
111
  target: "e111uita2",
113
112
  label: "AdditionalIcon"
@@ -117,17 +116,17 @@ var AdditionalIcon = /*#__PURE__*/(0, _base.default)("span", {
117
116
  } : {
118
117
  name: "1qitjsl",
119
118
  styles: "padding:1px;border:1px solid currentColor;border-radius:100%;font-size:15px;width:25px;text-align:center",
120
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA4KkC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
119
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA2KkC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
121
120
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
122
121
  });
123
122
  var StyledContentWrapper = /*#__PURE__*/(0, _base.default)("div", {
124
123
  target: "e111uita1",
125
124
  label: "StyledContentWrapper"
126
- })("padding-top:", _core.spacing.normal, ";border-left:6px solid ", _core.colors.brand.light, ";color:", _core.colors.text.primary, ";background-color:", _core.colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAqLsD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
125
+ })("padding-top:", _core.spacing.normal, ";border-left:6px solid ", _core.colors.brand.light, ";color:", _core.colors.text.primary, ";background-color:", _core.colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoLsD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
127
126
  var StyledModalHeader = /*#__PURE__*/(0, _base.default)(_modal.ModalHeader, {
128
127
  target: "e111uita0",
129
128
  label: "StyledModalHeader"
130
- })("padding:", _core.spacing.small, " ", _core.spacing.nsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA4L6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: string;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{renderMarkdown ? parse(renderMarkdown(introduction)) : introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
129
+ })("padding:", _core.spacing.small, " ", _core.spacing.nsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA2L6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-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 { ReactNode, MouseEvent, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  mask-image: radial-gradient(white, black);\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>{introduction}</TopicIntroduction>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
131
130
  var Topic = function Topic(_ref2) {
132
131
  var id = _ref2.id,
133
132
  title = _ref2.title,
@@ -136,7 +135,6 @@ var Topic = function Topic(_ref2) {
136
135
  subTopics = _ref2.subTopics,
137
136
  onSubTopicSelected = _ref2.onSubTopicSelected,
138
137
  isLoading = _ref2.isLoading,
139
- renderMarkdown = _ref2.renderMarkdown,
140
138
  invertedStyle = _ref2.invertedStyle,
141
139
  onToggleShowContent = _ref2.onToggleShowContent,
142
140
  showContent = _ref2.showContent,
@@ -195,7 +193,7 @@ var Topic = function Topic(_ref2) {
195
193
  })]
196
194
  })]
197
195
  }), (0, _jsxRuntime.jsx)(TopicIntroduction, {
198
- children: renderMarkdown ? (0, _htmlReactParser.default)(renderMarkdown(introduction)) : introduction
196
+ children: introduction
199
197
  })]
200
198
  }), metaImage && (0, _jsxRuntime.jsx)(TopicHeaderVisualElementWrapper, {
201
199
  children: (visualElementEmbedMeta === null || visualElementEmbedMeta === void 0 ? void 0 : visualElementEmbedMeta.status) === 'success' ? (0, _jsxRuntime.jsxs)(_modal.Modal, {