@ndla/ui 50.5.3 → 50.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (51) hide show
  1. package/es/FileList/File.js +9 -55
  2. package/es/FileList/FileList.js +19 -8
  3. package/es/FileList/Format.js +70 -0
  4. package/es/FileList/index.js +1 -0
  5. package/es/LicenseByline/EmbedByline.js +7 -7
  6. package/es/Navigation/NavigationBox.js +43 -78
  7. package/es/ResourceGroup/ResourceItem.js +15 -11
  8. package/es/SearchTypeResult/components/ItemContexts.js +9 -7
  9. package/es/Topic/Topic.js +24 -52
  10. package/es/all.css +1 -1
  11. package/es/index.js +1 -1
  12. package/es/locale/messages-en.js +1 -1
  13. package/es/locale/messages-nb.js +1 -1
  14. package/es/locale/messages-nn.js +1 -1
  15. package/es/locale/messages-sma.js +1 -1
  16. package/lib/FileList/File.js +11 -54
  17. package/lib/FileList/FileList.d.ts +4 -3
  18. package/lib/FileList/FileList.js +19 -8
  19. package/lib/FileList/Format.d.ts +16 -0
  20. package/lib/FileList/Format.js +74 -0
  21. package/lib/FileList/index.d.ts +1 -0
  22. package/lib/FileList/index.js +7 -0
  23. package/lib/LicenseByline/EmbedByline.js +7 -7
  24. package/lib/Navigation/NavigationBox.d.ts +2 -5
  25. package/lib/Navigation/NavigationBox.js +42 -77
  26. package/lib/ResourceGroup/ResourceItem.js +15 -11
  27. package/lib/SearchTypeResult/components/ItemContexts.js +9 -7
  28. package/lib/Topic/Topic.js +24 -50
  29. package/lib/all.css +1 -1
  30. package/lib/index.d.ts +1 -1
  31. package/lib/index.js +6 -0
  32. package/lib/locale/messages-en.js +1 -1
  33. package/lib/locale/messages-nb.js +1 -1
  34. package/lib/locale/messages-nn.js +1 -1
  35. package/lib/locale/messages-sma.js +1 -1
  36. package/package.json +12 -12
  37. package/src/FileList/File.tsx +5 -65
  38. package/src/FileList/FileList.tsx +17 -20
  39. package/src/FileList/Format.tsx +75 -0
  40. package/src/FileList/index.ts +1 -0
  41. package/src/LicenseByline/EmbedByline.tsx +2 -1
  42. package/src/Navigation/NavigationBox.tsx +76 -104
  43. package/src/ResourceGroup/ResourceItem.tsx +2 -2
  44. package/src/SearchTypeResult/components/ItemContexts.tsx +5 -1
  45. package/src/Topic/Topic.tsx +26 -39
  46. package/src/all.scss +0 -1
  47. package/src/index.ts +1 -1
  48. package/src/locale/messages-en.ts +1 -1
  49. package/src/locale/messages-nb.ts +1 -1
  50. package/src/locale/messages-nn.ts +1 -1
  51. package/src/locale/messages-sma.ts +1 -1
package/es/Topic/Topic.js CHANGED
@@ -1,5 +1,4 @@
1
1
  import _styled from "@emotion/styled/base";
2
- 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)."; }
3
2
  /**
4
3
  * Copyright (c) 2021-present, NDLA.
5
4
  *
@@ -12,9 +11,9 @@ import { useMemo } from 'react';
12
11
  import { useTranslation } from 'react-i18next';
13
12
  import { css } from '@emotion/react';
14
13
  import { ButtonV2 } from '@ndla/button';
15
- import { animations, breakpoints, colors, mq, spacing } from '@ndla/core';
14
+ import { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';
16
15
  import { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';
17
- import { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';
16
+ import { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';
18
17
  import { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';
19
18
  import { Text, Heading } from '@ndla/typography';
20
19
  import Loader from './Loader';
@@ -28,15 +27,15 @@ import { jsxs as _jsxs } from "@emotion/react/jsx-runtime";
28
27
  const Wrapper = /*#__PURE__*/_styled("div", {
29
28
  target: "e111uita12",
30
29
  label: "Wrapper"
31
- })("display:flex;flex-direction:column;gap:", spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+B0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
30
+ })("display:flex;flex-direction:column;gap:", spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA0B0B","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
32
31
  const frameStyle = /*#__PURE__*/css(mq.range({
33
32
  from: breakpoints.tabletWide
34
33
  }), "{padding:40px 40px;border:2px solid ", colors.brand.neutral7, ";}", mq.range({
35
34
  from: breakpoints.desktop
36
35
  }), "{padding:40px 80px;}", mq.range({
37
36
  from: '1180px'
38
- }), "{padding:60px 160px;};label:frameStyle;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAqCsB","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
39
- const _invertedStyle = /*#__PURE__*/css("color:", colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkD0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
37
+ }), "{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":"AAgCsB","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
38
+ const _invertedStyle = /*#__PURE__*/css("color:", colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA6C0B","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
40
39
  const TopicHeaderVisualElementWrapper = /*#__PURE__*/_styled("div", {
41
40
  target: "e111uita11",
42
41
  label: "TopicHeaderVisualElementWrapper"
@@ -44,79 +43,54 @@ const TopicHeaderVisualElementWrapper = /*#__PURE__*/_styled("div", {
44
43
  from: breakpoints.mobileWide
45
44
  }), "{width:150px;height:150px;}", mq.range({
46
45
  from: breakpoints.tabletWide
47
- }), "{width:200px;height:200px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAsDkD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
46
+ }), "{width:200px;height:200px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiDkD","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
48
47
  const ShowVisualElementWrapper = /*#__PURE__*/_styled("div", {
49
48
  target: "e111uita10",
50
49
  label: "ShowVisualElementWrapper"
51
- })(process.env.NODE_ENV === "production" ? {
52
- name: "dleg4l",
53
- 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)"
54
- } : {
55
- name: "dleg4l",
56
- 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)",
57
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoE2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */",
58
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
59
- });
50
+ })("border-radius:", misc.borderRadiusLarge, ";width:100%;height:100%;overflow:hidden;aspect-ratio:1;mask-image:radial-gradient(white, black);-webkit-mask-image:-webkit-radial-gradient(white, black);" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+D2C","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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
51
  const VisualElementButton = /*#__PURE__*/_styled(ButtonV2, {
61
52
  target: "e111uita9",
62
53
  label: "VisualElementButton"
63
- })("color:", colors.brand.secondary, ";width:100%;height:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA8E4C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
54
+ })("color:", 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":"AAyE4C","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
64
55
  const TopicHeaderImage = /*#__PURE__*/_styled("img", {
65
56
  target: "e111uita8",
66
57
  label: "TopicHeaderImage"
67
- })("border-radius:50%;aspect-ratio:1;width:100%;height:100%;object-fit:cover;transition:transform ", animations.durations.fast, ";", VisualElementButton, ":hover &{transform:scale(1.1);opacity:1.2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoFmC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
58
+ })("border-radius:", misc.borderRadiusLarge, ";aspect-ratio:1;width:100%;height:100%;object-fit:cover;transition:transform ", 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":"AA+EmC","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
68
59
  const ExpandVisualElementButton = /*#__PURE__*/_styled("span", {
69
60
  target: "e111uita7",
70
61
  label: "ExpandVisualElementButton"
71
62
  })("position:absolute;right:-10px;bottom:-4px;transition:all ", animations.durations.fast, ";svg{width:24px;height:24px;}", VisualElementButton, ":hover &{right:10px;}", mq.range({
72
63
  from: breakpoints.mobileWide
73
- }), "{right:0;bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiG6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
64
+ }), "{right:0;bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA4F6C","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
74
65
  const TopicHeaderOverlay = /*#__PURE__*/_styled("div", {
75
66
  target: "e111uita6",
76
67
  label: "TopicHeaderOverlay"
77
- })("background:black;opacity:0.05;position:absolute;top:0;left:0;bottom:0;right:0;border-radius:50%;transition:opacity ", animations.durations.fast, ";", VisualElementButton, ":hover &{opacity:0.1;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmHqC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
68
+ })("background:black;opacity:0.05;position:absolute;top:0;left:0;bottom:0;right:0;border-radius:", misc.borderRadiusLarge, ";transition:opacity ", 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":"AA8GqC","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
78
69
  const TopicIntroductionWrapper = /*#__PURE__*/_styled("div", {
79
70
  target: "e111uita5",
80
71
  label: "TopicIntroductionWrapper"
81
- })("display:flex;gap:", spacing.xsmall, ";justify-content:space-between;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkI2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
72
+ })("display:flex;gap:", 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":"AA6H2C","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
82
73
  const HeadingWrapper = /*#__PURE__*/_styled("hgroup", {
83
74
  target: "e111uita4",
84
75
  label: "HeadingWrapper"
85
- })("display:flex;flex-wrap:wrap;align-items:center;gap:", spacing.small, ";h1{margin:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwIoC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactNode, MouseEvent, useMemo } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\n        </div>\n        {metaImage && (\n          <TopicHeaderVisualElementWrapper>\n            {visualElementEmbedMeta?.status === 'success' ? (\n              <Modal>\n                <ModalTrigger>\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={visualElementEmbedMeta.resource === 'image' ? t('image.largeSize') : t('visualElement.show')}\n                  >\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${metaImage.url}?${makeSrcQueryString(800, metaImage.crop, metaImage.focalPoint)}`}\n                        alt={metaImage.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                </ModalTrigger>\n                <ModalContent\n                  aria-label={t('topicPage.imageModal')}\n                  animation=\"subtle\"\n                  animationDuration={50}\n                  size=\"large\"\n                >\n                  <StyledModalHeader>\n                    <ModalCloseButton />\n                  </StyledModalHeader>\n                  {visualElement}\n                </ModalContent>\n              </Modal>\n            ) : (\n              <TopicHeaderImage\n                src={`${metaImage.url}?${makeSrcQueryString(400, metaImage.crop, metaImage.focalPoint)}`}\n                alt={metaImage.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}\n            >\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
86
- var _ref = process.env.NODE_ENV === "production" ? {
87
- name: "1pu3m2l-StyledButtonWrapper",
88
- styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;"
89
- } : {
90
- name: "1pu3m2l-StyledButtonWrapper",
91
- styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;",
92
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwJO","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */",
93
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
94
- };
76
+ })("display:flex;flex-wrap:wrap;align-items:center;gap:", 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":"AAmIoC","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
95
77
  const StyledButtonWrapper = /*#__PURE__*/_styled("div", {
96
78
  target: "e111uita3",
97
79
  label: "StyledButtonWrapper"
98
- })("margin-top:", spacing.small, ";padding:", spacing.xsmall, " 0 ", spacing.xsmall, " ", spacing.medium, ";border-left:6px solid ", colors.brand.light, ";", props => props.invertedStyle && _ref, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkJqD","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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"]} */"));
99
- const AdditionalIcon = /*#__PURE__*/_styled("span", {
80
+ })("margin-top:", spacing.small, ";padding:", spacing.xsmall, " 0 ", spacing.xsmall, " ", spacing.medium, ";border-left:", spacing.xsmall, " solid ", colors.brand.light, ";&[data-inverted='true']{button{color:", colors.white, ";&:hover,&:focus{color:", colors.white, ";}}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA6IsC","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
81
+ const StyledAdditional = /*#__PURE__*/_styled(Additional, {
100
82
  target: "e111uita2",
101
- label: "AdditionalIcon"
102
- })(process.env.NODE_ENV === "production" ? {
103
- name: "1qitjsl",
104
- styles: "padding:1px;border:1px solid currentColor;border-radius:100%;font-size:15px;width:25px;text-align:center"
105
- } : {
106
- name: "1qitjsl",
107
- styles: "padding:1px;border:1px solid currentColor;border-radius:100%;font-size:15px;width:25px;text-align:center",
108
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmKkC","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
110
- });
83
+ label: "StyledAdditional"
84
+ })("color:", colors.brand.dark, ";height:", spacing.normal, ";width:", spacing.normal, ";padding:1px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA4J2C","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
111
85
  const StyledContentWrapper = /*#__PURE__*/_styled("div", {
112
86
  target: "e111uita1",
113
87
  label: "StyledContentWrapper"
114
- })("padding-top:", spacing.normal, ";border-left:6px solid ", colors.brand.light, ";color:", colors.text.primary, ";background-color:", colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA4KsD","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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
+ })("background-color:", colors.white, ";border-left:", spacing.xsmall, " solid ", colors.brand.light, ";color:", colors.text.primary, ";padding-top:", spacing.normal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmKuC","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
115
89
  const StyledModalHeader = /*#__PURE__*/_styled(ModalHeader, {
116
90
  target: "e111uita0",
117
91
  label: "StyledModalHeader"
118
- })("padding:", spacing.small, " ", spacing.nsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmL6C","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\n\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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 StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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
- const Topic = _ref2 => {
92
+ })("padding:", spacing.small, " ", spacing.nsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA0K6C","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 { useTranslation } from 'react-i18next';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { ButtonV2 } from '@ndla/button';\nimport { animations, breakpoints, colors, misc, mq, spacing } from '@ndla/core';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { Additional, ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalContent, Modal, ModalHeader, ModalTrigger } from '@ndla/modal';\nimport { EmbedMetaData } from '@ndla/types-embed';\nimport { Text, Heading } from '@ndla/typography';\nimport Loader from './Loader';\nimport { getCrop, getFocalPoint } from '../Embed/ImageEmbed';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\nimport { NavigationBox } from '../Navigation';\nimport { ItemProps } from '../Navigation/NavigationBox';\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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: ${misc.borderRadiusLarge};\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 StyledButtonWrapper = styled.div`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  &[data-inverted='true'] {\n    button {\n      color: ${colors.white};\n      &:hover,\n      &:focus {\n        color: ${colors.white};\n      }\n    }\n  }\n`;\n\nconst StyledAdditional = styled(Additional)`\n  color: ${colors.brand.dark};\n  height: ${spacing.normal};\n  width: ${spacing.normal};\n  padding: 1px;\n`;\n\nconst StyledContentWrapper = styled.div`\n  background-color: ${colors.white};\n  border-left: ${spacing.xsmall} solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  padding-top: ${spacing.normal};\n`;\n\nconst StyledModalHeader = styled(ModalHeader)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nexport type TopicProps = {\n  id?: string;\n  metaImage?: {\n    url: string;\n    alt: string;\n  };\n  title: string;\n  introduction: ReactNode;\n  resources?: ReactNode;\n  visualElementEmbedMeta?: EmbedMetaData;\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n  visualElement?: ReactNode;\n};\n\ninterface MetaImageType {\n  url: string;\n  alt: string;\n  crop?: ImageCrop;\n  focalPoint?: ImageFocalPoint;\n}\n\nconst Topic = ({\n  id,\n  title,\n  introduction,\n  resources,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  metaImage: articleMetaImage,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  visualElementEmbedMeta,\n  children,\n  visualElement,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n\n  const VisualElementIcon = useMemo(() => {\n    if (!visualElementEmbedMeta || visualElementEmbedMeta.status === 'error') return null;\n    else if (visualElementEmbedMeta.resource === 'brightcove') {\n      return PlayCircleFilled;\n    } else if (visualElementEmbedMeta.resource === 'image') {\n      return ExpandTwoArrows;\n    } else return CursorClick;\n  }, [visualElementEmbedMeta]);\n\n  const metaImage: MetaImageType | undefined = useMemo(() => {\n    if (visualElementEmbedMeta?.resource === 'image' && visualElementEmbedMeta.status === 'success') {\n      return {\n        url: visualElementEmbedMeta.data.image?.imageUrl,\n        alt: visualElementEmbedMeta.data.alttext?.alttext,\n        crop: getCrop(visualElementEmbedMeta.embedData),\n        focalPoint: getFocalPoint(visualElementEmbedMeta.embedData),\n      };\n    } else return articleMetaImage;\n  }, [articleMetaImage, visualElementEmbedMeta]);\n\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading) {\n    return <Wrapper css={wrapperStyle}>{isLoading ? <Loader /> : null}</Wrapper>;\n  }\n\n  return (\n    <Wrapper css={wrapperStyle}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <Heading element=\"h1\" headingStyle=\"h2\" id={id} tabIndex={-1}>\n              {title}\n            </Heading>\n            {isAdditionalTopic && (\n              <>\n                <StyledAdditional aria-hidden=\"true\" />\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <Text textStyle=\"ingress\" element=\"div\">\n            {introduction}\n          </Text>\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 data-inverted={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 && <StyledContentWrapper id={contentId}>{children}</StyledContentWrapper>}\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"]} */"));
93
+ const Topic = _ref => {
120
94
  let {
121
95
  id,
122
96
  title,
@@ -135,7 +109,7 @@ const Topic = _ref2 => {
135
109
  visualElementEmbedMeta,
136
110
  children,
137
111
  visualElement
138
- } = _ref2;
112
+ } = _ref;
139
113
  const {
140
114
  t
141
115
  } = useTranslation();
@@ -176,9 +150,8 @@ const Topic = _ref2 => {
176
150
  tabIndex: -1,
177
151
  children: title
178
152
  }), isAdditionalTopic && _jsxs(_Fragment, {
179
- children: [_jsx(AdditionalIcon, {
180
- "aria-hidden": "true",
181
- children: "T"
153
+ children: [_jsx(StyledAdditional, {
154
+ "aria-hidden": "true"
182
155
  }), _jsx("span", {
183
156
  children: t('navigation.additionalTopic')
184
157
  })]
@@ -221,7 +194,7 @@ const Topic = _ref2 => {
221
194
  children: messageBox
222
195
  }), _jsxs("div", {
223
196
  children: [onToggleShowContent && _jsx(StyledButtonWrapper, {
224
- invertedStyle: invertedStyle,
197
+ "data-inverted": invertedStyle,
225
198
  children: _jsx(ButtonV2, {
226
199
  "aria-expanded": !!showContent,
227
200
  "aria-controls": contentId,
@@ -235,7 +208,6 @@ const Topic = _ref2 => {
235
208
  })
236
209
  }), showContent && _jsx(StyledContentWrapper, {
237
210
  id: contentId,
238
- invertedStyle: invertedStyle,
239
211
  children: children
240
212
  })]
241
213
  }), subTopics && subTopics.length !== 0 && _jsx(NavigationBox, {