@ndla/ui 43.0.3 → 44.0.1

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 (53) hide show
  1. package/es/Article/Article.js +6 -14
  2. package/es/Article/ArticleNotions.js +24 -48
  3. package/es/Article/index.js +1 -2
  4. package/es/ContactBlock/ContactBlock.js +13 -13
  5. package/es/Embed/ImageEmbed.js +2 -2
  6. package/es/Grid/Grid.js +15 -2
  7. package/es/Grid/index.js +1 -1
  8. package/es/LinkBlock/LinkBlock.js +16 -6
  9. package/es/{Article/ArticleContent.js → LinkBlock/LinkBlockSection.js} +22 -30
  10. package/es/LinkBlock/index.js +2 -1
  11. package/es/Topic/Topic.js +60 -42
  12. package/es/index.js +4 -3
  13. package/lib/Article/Article.d.ts +3 -9
  14. package/lib/Article/Article.js +6 -14
  15. package/lib/Article/ArticleNotions.d.ts +3 -9
  16. package/lib/Article/ArticleNotions.js +30 -56
  17. package/lib/Article/index.d.ts +1 -2
  18. package/lib/Article/index.js +0 -7
  19. package/lib/ContactBlock/ContactBlock.js +13 -13
  20. package/lib/Embed/ImageEmbed.d.ts +11 -1
  21. package/lib/Embed/ImageEmbed.js +3 -1
  22. package/lib/Grid/Grid.d.ts +4 -0
  23. package/lib/Grid/Grid.js +23 -10
  24. package/lib/Grid/index.d.ts +1 -1
  25. package/lib/Grid/index.js +10 -2
  26. package/lib/LinkBlock/LinkBlock.d.ts +2 -7
  27. package/lib/LinkBlock/LinkBlock.js +16 -6
  28. package/lib/LinkBlock/LinkBlockSection.d.ts +13 -0
  29. package/lib/{Article/ArticleContent.js → LinkBlock/LinkBlockSection.js} +22 -27
  30. package/lib/LinkBlock/index.d.ts +1 -0
  31. package/lib/LinkBlock/index.js +7 -0
  32. package/lib/Topic/Topic.d.ts +10 -17
  33. package/lib/Topic/Topic.js +62 -44
  34. package/lib/index.d.ts +3 -2
  35. package/lib/index.js +19 -6
  36. package/lib/types.d.ts +1 -1
  37. package/package.json +6 -5
  38. package/src/Article/Article.tsx +6 -16
  39. package/src/Article/ArticleNotions.tsx +9 -78
  40. package/src/Article/index.ts +1 -10
  41. package/src/ContactBlock/ContactBlock.tsx +3 -0
  42. package/src/Embed/ImageEmbed.tsx +2 -2
  43. package/src/Grid/Grid.tsx +9 -1
  44. package/src/Grid/index.ts +1 -1
  45. package/src/LinkBlock/LinkBlock.stories.tsx +18 -8
  46. package/src/LinkBlock/LinkBlock.tsx +12 -9
  47. package/src/LinkBlock/LinkBlockSection.tsx +42 -0
  48. package/src/LinkBlock/index.ts +1 -0
  49. package/src/Topic/Topic.tsx +58 -39
  50. package/src/index.ts +3 -2
  51. package/src/types.ts +1 -1
  52. package/lib/Article/ArticleContent.d.ts +0 -7
  53. package/src/Article/ArticleContent.tsx +0 -40
@@ -5,19 +5,21 @@ Object.defineProperty(exports, "__esModule", {
5
5
  });
6
6
  exports.default = void 0;
7
7
  var _base = _interopRequireDefault(require("@emotion/styled/base"));
8
+ var _react = require("react");
8
9
  var _core = require("@ndla/core");
9
10
  var _htmlReactParser = _interopRequireDefault(require("html-react-parser"));
10
11
  var _common = require("@ndla/icons/common");
11
12
  var _modal = require("@ndla/modal");
12
13
  var _button = require("@ndla/button");
13
14
  var _action = require("@ndla/icons/action");
14
- var _react = require("@emotion/react");
15
+ var _react2 = require("@emotion/react");
15
16
  var _reactI18next = require("react-i18next");
16
17
  var _Loader = _interopRequireDefault(require("./Loader"));
17
18
  var _Navigation = require("../Navigation");
18
19
  var _Image = require("../Image");
19
20
  var _Messages = require("../Messages");
20
21
  var _Typography = require("../Typography");
22
+ var _ImageEmbed = require("../Embed/ImageEmbed");
21
23
  var _jsxRuntime = require("@emotion/react/jsx-runtime");
22
24
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
23
25
  function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; } /**
@@ -30,15 +32,15 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
30
32
  var Wrapper = /*#__PURE__*/(0, _base.default)("div", {
31
33
  target: "e111uita13",
32
34
  label: "Wrapper"
33
- })("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":"AA8B0B","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
34
- var frameStyle = /*#__PURE__*/(0, _react.css)(_core.mq.range({
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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"]} */"));
36
+ var frameStyle = /*#__PURE__*/(0, _react2.css)(_core.mq.range({
35
37
  from: _core.breakpoints.tabletWide
36
38
  }), "{padding:40px 40px;border:2px solid ", _core.colors.brand.neutral7, ";}", _core.mq.range({
37
39
  from: _core.breakpoints.desktop
38
40
  }), "{padding:40px 80px;}", _core.mq.range({
39
41
  from: '1180px'
40
- }), "{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":"AAoCsB","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
41
- var _invertedStyle = /*#__PURE__*/(0, _react.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":"AAiD0B","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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"]} */"));
42
44
  var TopicHeaderVisualElementWrapper = /*#__PURE__*/(0, _base.default)("div", {
43
45
  target: "e111uita12",
44
46
  label: "TopicHeaderVisualElementWrapper"
@@ -46,58 +48,58 @@ var TopicHeaderVisualElementWrapper = /*#__PURE__*/(0, _base.default)("div", {
46
48
  from: _core.breakpoints.mobileWide
47
49
  }), "{width:150px;height:150px;}", _core.mq.range({
48
50
  from: _core.breakpoints.tabletWide
49
- }), "{width:200px;height:200px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAqDkD","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
52
  var ShowVisualElementWrapper = /*#__PURE__*/(0, _base.default)("div", {
51
53
  target: "e111uita11",
52
54
  label: "ShowVisualElementWrapper"
53
55
  })(process.env.NODE_ENV === "production" ? {
54
- name: "1v6dfx3",
55
- styles: "border-radius:50%;width:100%;height:100%;overflow:hidden;aspect-ratio:1;-webkit-mask-image:-webkit-radial-gradient(white, black)"
56
+ name: "dleg4l",
57
+ 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)"
56
58
  } : {
57
- name: "1v6dfx3",
58
- styles: "border-radius:50%;width:100%;height:100%;overflow:hidden;aspect-ratio:1;-webkit-mask-image:-webkit-radial-gradient(white, black)",
59
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmE2C","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
59
+ name: "dleg4l",
60
+ 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
62
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
61
63
  });
62
64
  var VisualElementButton = /*#__PURE__*/(0, _base.default)(_button.ButtonV2, {
63
65
  target: "e111uita10",
64
66
  label: "VisualElementButton"
65
- })("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":"AA4E4C","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
68
  var TopicHeaderImage = /*#__PURE__*/(0, _base.default)("img", {
67
69
  target: "e111uita9",
68
70
  label: "TopicHeaderImage"
69
- })("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":"AAkFmC","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
72
  var ExpandVisualElementButton = /*#__PURE__*/(0, _base.default)("span", {
71
73
  target: "e111uita8",
72
74
  label: "ExpandVisualElementButton"
73
75
  })("position:absolute;right:-10px;bottom:-4px;transition:all ", _core.animations.durations.fast, ";svg{width:24px;height:24px;}", VisualElementButton, ":hover &{right:10px;}", _core.mq.range({
74
76
  from: _core.breakpoints.mobileWide
75
- }), "{right:0;bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+F6C","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
78
  var TopicHeaderOverlay = /*#__PURE__*/(0, _base.default)("div", {
77
79
  target: "e111uita7",
78
80
  label: "TopicHeaderOverlay"
79
- })("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":"AAiHqC","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
82
  var TopicIntroductionWrapper = /*#__PURE__*/(0, _base.default)("div", {
81
83
  target: "e111uita6",
82
84
  label: "TopicIntroductionWrapper"
83
- })("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":"AAgI2C","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
86
  var HeadingWrapper = /*#__PURE__*/(0, _base.default)("hgroup", {
85
87
  target: "e111uita5",
86
88
  label: "HeadingWrapper"
87
- })("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":"AAsIoC","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
90
  var TopicIntroduction = /*#__PURE__*/(0, _base.default)("div", {
89
91
  target: "e111uita4",
90
92
  label: "TopicIntroduction"
91
93
  })("font-weight:", _core.fonts.weight.light, ";max-width:612px;", _core.mq.range({
92
94
  from: _core.breakpoints.tablet
93
- }), "{", _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":"AAgJoC","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
96
  var _ref = process.env.NODE_ENV === "production" ? {
95
97
  name: "1pu3m2l-StyledButtonWrapper",
96
98
  styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;"
97
99
  } : {
98
100
  name: "1pu3m2l-StyledButtonWrapper",
99
101
  styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;",
100
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA8JO","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
103
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
102
104
  };
103
105
  var StyledButtonWrapper = /*#__PURE__*/(0, _base.default)("div", {
@@ -105,7 +107,7 @@ var StyledButtonWrapper = /*#__PURE__*/(0, _base.default)("div", {
105
107
  label: "StyledButtonWrapper"
106
108
  })("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) {
107
109
  return props.invertedStyle && _ref;
108
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwJqD","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
111
  var AdditionalIcon = /*#__PURE__*/(0, _base.default)("span", {
110
112
  target: "e111uita2",
111
113
  label: "AdditionalIcon"
@@ -115,26 +117,22 @@ var AdditionalIcon = /*#__PURE__*/(0, _base.default)("span", {
115
117
  } : {
116
118
  name: "1qitjsl",
117
119
  styles: "padding:1px;border:1px solid currentColor;border-radius:100%;font-size:15px;width:25px;text-align:center",
118
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAyKkC","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
121
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
120
122
  });
121
123
  var StyledContentWrapper = /*#__PURE__*/(0, _base.default)("div", {
122
124
  target: "e111uita1",
123
125
  label: "StyledContentWrapper"
124
- })("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":"AAkLsD","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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
127
  var StyledModalHeader = /*#__PURE__*/(0, _base.default)(_modal.ModalHeader, {
126
128
  target: "e111uita0",
127
129
  label: "StyledModalHeader"
128
- })("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":"AAyL6C","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, ComponentType } 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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  -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\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\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};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {topic.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>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.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                  {topic.visualElement && topic.visualElement.element}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.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      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
129
- var icons = {
130
- image: _action.ExpandTwoArrows,
131
- video: _common.PlayCircleFilled,
132
- other: _action.CursorClick
133
- };
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 Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { Heading } from '../Typography';\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  const testId = 'nav-topic-about';\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 (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\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"]} */"));
134
131
  var Topic = function Topic(_ref2) {
135
- var _topic$visualElement;
136
132
  var id = _ref2.id,
137
- topic = _ref2.topic,
133
+ title = _ref2.title,
134
+ introduction = _ref2.introduction,
135
+ resources = _ref2.resources,
138
136
  subTopics = _ref2.subTopics,
139
137
  onSubTopicSelected = _ref2.onSubTopicSelected,
140
138
  isLoading = _ref2.isLoading,
@@ -142,17 +140,37 @@ var Topic = function Topic(_ref2) {
142
140
  invertedStyle = _ref2.invertedStyle,
143
141
  onToggleShowContent = _ref2.onToggleShowContent,
144
142
  showContent = _ref2.showContent,
143
+ articleMetaImage = _ref2.metaImage,
145
144
  isAdditionalTopic = _ref2.isAdditionalTopic,
146
145
  frame = _ref2.frame,
147
146
  messageBox = _ref2.messageBox,
148
- children = _ref2.children;
147
+ visualElementEmbedMeta = _ref2.visualElementEmbedMeta,
148
+ children = _ref2.children,
149
+ visualElement = _ref2.visualElement;
149
150
  var _useTranslation = (0, _reactI18next.useTranslation)(),
150
151
  t = _useTranslation.t;
151
152
  var contentId = "expanded-description-".concat(id);
152
153
  var testId = 'nav-topic-about';
153
- var VisualElementIcon = topic !== null && topic !== void 0 && (_topic$visualElement = topic.visualElement) !== null && _topic$visualElement !== void 0 && _topic$visualElement.type ? icons[topic.visualElement.type] : null;
154
+ var VisualElementIcon = (0, _react.useMemo)(function () {
155
+ if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;else if (visualElementEmbedMeta.resource === 'brightcove') {
156
+ return _common.PlayCircleFilled;
157
+ } else if (visualElementEmbedMeta.resource === 'image') {
158
+ return _action.ExpandTwoArrows;
159
+ } else return _action.CursorClick;
160
+ }, [visualElementEmbedMeta]);
161
+ var metaImage = (0, _react.useMemo)(function () {
162
+ if ((visualElementEmbedMeta === null || visualElementEmbedMeta === void 0 ? void 0 : visualElementEmbedMeta.resource) === 'image' && visualElementEmbedMeta.status === 'success') {
163
+ var _visualElementEmbedMe, _visualElementEmbedMe2;
164
+ return {
165
+ url: (_visualElementEmbedMe = visualElementEmbedMeta.data.image) === null || _visualElementEmbedMe === void 0 ? void 0 : _visualElementEmbedMe.imageUrl,
166
+ alt: (_visualElementEmbedMe2 = visualElementEmbedMeta.data.alttext) === null || _visualElementEmbedMe2 === void 0 ? void 0 : _visualElementEmbedMe2.alttext,
167
+ crop: (0, _ImageEmbed.getCrop)(visualElementEmbedMeta.embedData),
168
+ focalPoint: (0, _ImageEmbed.getFocalPoint)(visualElementEmbedMeta.embedData)
169
+ };
170
+ } else return articleMetaImage;
171
+ }, [articleMetaImage, visualElementEmbedMeta]);
154
172
  var wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];
155
- if (isLoading || !topic) {
173
+ if (isLoading) {
156
174
  return (0, _jsxRuntime.jsx)(Wrapper, {
157
175
  css: wrapperStyle,
158
176
  "data-testid": testId,
@@ -170,7 +188,7 @@ var Topic = function Topic(_ref2) {
170
188
  headingStyle: "h2",
171
189
  id: id,
172
190
  tabIndex: -1,
173
- children: topic.title
191
+ children: title
174
192
  }), isAdditionalTopic && (0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
175
193
  children: [(0, _jsxRuntime.jsx)(AdditionalIcon, {
176
194
  "aria-hidden": "true",
@@ -180,18 +198,18 @@ var Topic = function Topic(_ref2) {
180
198
  })]
181
199
  })]
182
200
  }), (0, _jsxRuntime.jsx)(TopicIntroduction, {
183
- children: renderMarkdown ? (0, _htmlReactParser.default)(renderMarkdown(topic.introduction)) : topic.introduction
201
+ children: renderMarkdown ? (0, _htmlReactParser.default)(renderMarkdown(introduction)) : introduction
184
202
  })]
185
- }), topic.image && (0, _jsxRuntime.jsx)(TopicHeaderVisualElementWrapper, {
186
- children: topic.visualElement ? (0, _jsxRuntime.jsxs)(_modal.Modal, {
203
+ }), metaImage && (0, _jsxRuntime.jsx)(TopicHeaderVisualElementWrapper, {
204
+ children: (visualElementEmbedMeta === null || visualElementEmbedMeta === void 0 ? void 0 : visualElementEmbedMeta.status) === 'success' ? (0, _jsxRuntime.jsxs)(_modal.Modal, {
187
205
  children: [(0, _jsxRuntime.jsx)(_modal.ModalTrigger, {
188
206
  children: (0, _jsxRuntime.jsxs)(VisualElementButton, {
189
207
  variant: "stripped",
190
- title: topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show'),
208
+ title: visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show'),
191
209
  children: [(0, _jsxRuntime.jsxs)(ShowVisualElementWrapper, {
192
210
  children: [(0, _jsxRuntime.jsx)(TopicHeaderImage, {
193
- src: "".concat(topic.image.url, "?").concat((0, _Image.makeSrcQueryString)(800, topic.image.crop, topic.image.focalPoint)),
194
- alt: topic.image.alt
211
+ src: "".concat(metaImage.url, "?").concat((0, _Image.makeSrcQueryString)(800, metaImage.crop, metaImage.focalPoint)),
212
+ alt: metaImage.alt
195
213
  }), (0, _jsxRuntime.jsx)(TopicHeaderOverlay, {})]
196
214
  }), (0, _jsxRuntime.jsx)(ExpandVisualElementButton, {
197
215
  children: VisualElementIcon && (0, _jsxRuntime.jsx)(VisualElementIcon, {})
@@ -204,11 +222,11 @@ var Topic = function Topic(_ref2) {
204
222
  size: "large",
205
223
  children: [(0, _jsxRuntime.jsx)(StyledModalHeader, {
206
224
  children: (0, _jsxRuntime.jsx)(_modal.ModalCloseButton, {})
207
- }), topic.visualElement && topic.visualElement.element]
225
+ }), visualElement]
208
226
  })]
209
227
  }) : (0, _jsxRuntime.jsx)(TopicHeaderImage, {
210
- src: "".concat(topic.image.url, "?").concat((0, _Image.makeSrcQueryString)(400, topic.image.crop, topic.image.focalPoint)),
211
- alt: topic.image.alt
228
+ src: "".concat(metaImage.url, "?").concat((0, _Image.makeSrcQueryString)(400, metaImage.crop, metaImage.focalPoint)),
229
+ alt: metaImage.alt
212
230
  })
213
231
  })]
214
232
  }), messageBox && (0, _jsxRuntime.jsx)(_Messages.MessageBox, {
@@ -240,7 +258,7 @@ var Topic = function Topic(_ref2) {
240
258
  items: subTopics,
241
259
  onClick: onSubTopicSelected,
242
260
  invertedStyle: invertedStyle
243
- }), topic.resources]
261
+ }), resources]
244
262
  });
245
263
  };
246
264
  var _default = Topic;