@ndla/ui 50.1.6 → 50.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (84) hide show
  1. package/es/Embed/ConceptEmbed.js +28 -14
  2. package/es/Embed/FootnoteEmbed.js +7 -2
  3. package/es/Embed/conceptComponents.js +14 -10
  4. package/es/Filter/FilterCarousel.js +4 -4
  5. package/es/FrontpageArticle/FrontpageArticle.js +1 -1
  6. package/es/Gloss/Gloss.js +39 -34
  7. package/es/Gloss/GlossExample.js +62 -0
  8. package/es/Gloss/index.js +2 -1
  9. package/es/Resource/resourceComponents.js +11 -11
  10. package/es/ResourceGroup/ResourceItem.js +9 -9
  11. package/es/SearchTypeResult/ActiveFilters.js +6 -6
  12. package/es/TreeStructure/AddFolderButton.js +2 -2
  13. package/es/TreeStructure/FolderItem.js +7 -7
  14. package/es/TreeStructure/FolderItems.js +2 -2
  15. package/es/all.css +1 -1
  16. package/es/index.js +1 -1
  17. package/es/locale/messages-en.js +10 -5
  18. package/es/locale/messages-nb.js +10 -4
  19. package/es/locale/messages-nn.js +10 -4
  20. package/es/locale/messages-se.js +10 -4
  21. package/es/locale/messages-sma.js +13 -7
  22. package/es/model/index.js +1 -3
  23. package/lib/Embed/ConceptEmbed.d.ts +6 -2
  24. package/lib/Embed/ConceptEmbed.js +28 -14
  25. package/lib/Embed/FootnoteEmbed.js +8 -2
  26. package/lib/Embed/conceptComponents.d.ts +2 -0
  27. package/lib/Embed/conceptComponents.js +14 -10
  28. package/lib/Filter/FilterCarousel.js +4 -4
  29. package/lib/FrontpageArticle/FrontpageArticle.js +1 -1
  30. package/lib/Gloss/Gloss.d.ts +5 -14
  31. package/lib/Gloss/Gloss.js +39 -34
  32. package/lib/Gloss/GlossExample.d.ts +16 -0
  33. package/lib/Gloss/GlossExample.js +68 -0
  34. package/lib/Gloss/index.d.ts +1 -0
  35. package/lib/Gloss/index.js +7 -0
  36. package/lib/Resource/resourceComponents.js +11 -11
  37. package/lib/ResourceGroup/ResourceItem.js +9 -9
  38. package/lib/SearchTypeResult/ActiveFilters.js +6 -6
  39. package/lib/TreeStructure/AddFolderButton.js +2 -2
  40. package/lib/TreeStructure/FolderItem.js +7 -7
  41. package/lib/TreeStructure/FolderItems.js +2 -2
  42. package/lib/all.css +1 -1
  43. package/lib/index.d.ts +1 -1
  44. package/lib/index.js +6 -0
  45. package/lib/locale/messages-en.d.ts +8 -2
  46. package/lib/locale/messages-en.js +11 -7
  47. package/lib/locale/messages-nb.d.ts +7 -1
  48. package/lib/locale/messages-nb.js +10 -4
  49. package/lib/locale/messages-nn.d.ts +7 -1
  50. package/lib/locale/messages-nn.js +10 -4
  51. package/lib/locale/messages-se.d.ts +7 -1
  52. package/lib/locale/messages-se.js +10 -4
  53. package/lib/locale/messages-sma.d.ts +7 -1
  54. package/lib/locale/messages-sma.js +13 -7
  55. package/lib/model/index.d.ts +0 -2
  56. package/lib/model/index.js +1 -3
  57. package/package.json +5 -5
  58. package/src/Embed/ConceptEmbed.tsx +18 -0
  59. package/src/Embed/FootnoteEmbed.tsx +29 -3
  60. package/src/Embed/conceptComponents.tsx +6 -0
  61. package/src/Filter/FilterCarousel.tsx +3 -1
  62. package/src/FrontpageArticle/FrontpageArticle.tsx +1 -1
  63. package/src/Gloss/Gloss.tsx +41 -57
  64. package/src/Gloss/GlossExample.tsx +75 -0
  65. package/src/Gloss/index.tsx +1 -0
  66. package/src/Resource/resourceComponents.tsx +1 -1
  67. package/src/ResourceGroup/ResourceItem.tsx +3 -1
  68. package/src/SearchTypeResult/ActiveFilters.tsx +1 -3
  69. package/src/TreeStructure/AddFolderButton.tsx +2 -2
  70. package/src/TreeStructure/FolderItem.tsx +1 -1
  71. package/src/TreeStructure/FolderItems.tsx +2 -2
  72. package/src/index.ts +1 -1
  73. package/src/locale/messages-en.ts +9 -2
  74. package/src/locale/messages-nb.ts +9 -2
  75. package/src/locale/messages-nn.ts +9 -2
  76. package/src/locale/messages-se.ts +9 -2
  77. package/src/locale/messages-sma.ts +12 -5
  78. package/src/main.scss +0 -1
  79. package/src/model/index.ts +0 -2
  80. package/es/model/Transcriptions.js +0 -1
  81. package/lib/model/Transcriptions.d.ts +0 -11
  82. package/lib/model/Transcriptions.js +0 -5
  83. package/src/Article/component.footnotes.scss +0 -74
  84. package/src/model/Transcriptions.ts +0 -12
@@ -38,17 +38,17 @@ const PopoverWrapper = /*#__PURE__*/_styled("div", {
38
38
  return top;
39
39
  }, "px!important;}", mq.range({
40
40
  until: breakpoints.tablet
41
- }), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:100!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAiCkD","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
41
+ }), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:100!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAiCkD","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
42
42
  const ImageWrapper = /*#__PURE__*/_styled("div", {
43
43
  target: "e6acljj5",
44
44
  label: "ImageWrapper"
45
45
  })("float:right;padding-left:", spacing.normal, ";position:relative;", mq.range({
46
46
  until: breakpoints.tabletWide
47
- }), "{width:100%;padding-left:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA0D+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
47
+ }), "{width:100%;padding-left:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA0D+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
48
48
  const StyledButton = /*#__PURE__*/_styled("button", {
49
49
  target: "e6acljj4",
50
50
  label: "StyledButton"
51
- })("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;color:#000;position:relative;cursor:pointer;&:focus,&:hover{color:", colors.brand.primary, ";outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA4EkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
51
+ })("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;color:#000;position:relative;cursor:pointer;&:focus,&:hover{color:", colors.brand.primary, ";outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA4EkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
52
52
  export const ConceptEmbed = _ref2 => {
53
53
  let {
54
54
  embed,
@@ -90,7 +90,9 @@ export const ConceptEmbed = _ref2 => {
90
90
  }),
91
91
  conceptType: concept.conceptType,
92
92
  glossData: concept.glossData,
93
- lang: lang
93
+ lang: lang,
94
+ exampleIds: embed.embedData.exampleIds,
95
+ exampleLangs: embed.embedData.exampleLangs
94
96
  });
95
97
  } else if (embed.embedData.type === 'inline') {
96
98
  return _jsx(InlineConcept, {
@@ -107,7 +109,9 @@ export const ConceptEmbed = _ref2 => {
107
109
  }),
108
110
  conceptType: concept.conceptType,
109
111
  glossData: concept.glossData,
110
- lang: lang
112
+ lang: lang,
113
+ exampleIds: embed.embedData.exampleIds,
114
+ exampleLangs: embed.embedData.exampleLangs
111
115
  });
112
116
  } else {
113
117
  return _jsx(ConceptNotionV2, {
@@ -123,7 +127,9 @@ export const ConceptEmbed = _ref2 => {
123
127
  }),
124
128
  conceptType: concept.conceptType,
125
129
  glossData: concept.glossData,
126
- lang: lang
130
+ lang: lang,
131
+ exampleIds: embed.embedData.exampleIds,
132
+ exampleLangs: embed.embedData.exampleLangs
127
133
  });
128
134
  }
129
135
  };
@@ -136,19 +142,19 @@ const BaselineIcon = /*#__PURE__*/_styled("span", {
136
142
  } : {
137
143
  name: "6mki3j",
138
144
  styles: "display:block;border-bottom:5px double currentColor",
139
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA0KgC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
145
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAkLgC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
140
146
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
141
147
  });
142
148
  const NotionButton = /*#__PURE__*/_styled("button", {
143
149
  target: "e6acljj2",
144
150
  label: "NotionButton"
145
- })("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;position:relative;text-align:left;display:inline;color:", colors.notion.dark, ";cursor:pointer;&:focus,&:hover{background-color:", colors.notion.dark, ";color:", colors.white, ";outline:none;", BaselineIcon, "{border-color:transparent;}}&:active{color:", colors.notion.dark, ";background-color:", colors.notion.light, ";", BaselineIcon, "{border-color:currentColor;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA+KkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
151
+ })("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;position:relative;text-align:left;display:inline;color:", colors.notion.dark, ";cursor:pointer;&:focus,&:hover{background-color:", colors.notion.dark, ";color:", colors.white, ";outline:none;", BaselineIcon, "{border-color:transparent;}}&:active{color:", colors.notion.dark, ";background-color:", colors.notion.light, ";", BaselineIcon, "{border-color:currentColor;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAuLkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
146
152
  const StyledAnchor = /*#__PURE__*/_styled(Anchor, {
147
153
  target: "e6acljj1",
148
154
  label: "StyledAnchor"
149
155
  })(mq.range({
150
156
  until: breakpoints.tablet
151
- }), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAgNmC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
157
+ }), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAwNmC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
152
158
  const StyledAnchorSpan = /*#__PURE__*/_styled("span", {
153
159
  target: "e6acljj0",
154
160
  label: "StyledAnchorSpan"
@@ -158,7 +164,7 @@ const StyledAnchorSpan = /*#__PURE__*/_styled("span", {
158
164
  } : {
159
165
  name: "zcodsq",
160
166
  styles: "position:absolute;left:50%;align-self:center",
161
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAuNoC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
167
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA+NoC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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 { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
162
168
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
163
169
  });
164
170
  const getModalPosition = anchor => {
@@ -180,7 +186,9 @@ export const InlineConcept = _ref3 => {
180
186
  glossData,
181
187
  conceptType,
182
188
  headerButtons,
183
- lang
189
+ lang,
190
+ exampleIds,
191
+ exampleLangs
184
192
  } = _ref3;
185
193
  const {
186
194
  t
@@ -238,7 +246,9 @@ export const InlineConcept = _ref3 => {
238
246
  })
239
247
  }),
240
248
  conceptType: conceptType,
241
- glossData: glossData
249
+ glossData: glossData,
250
+ exampleIds: exampleIds,
251
+ exampleLangs: exampleLangs
242
252
  })
243
253
  })
244
254
  })
@@ -258,7 +268,9 @@ export const BlockConcept = _ref4 => {
258
268
  conceptHeartButton,
259
269
  glossData,
260
270
  conceptType,
261
- lang
271
+ lang,
272
+ exampleIds,
273
+ exampleLangs
262
274
  } = _ref4;
263
275
  const {
264
276
  t
@@ -354,7 +366,9 @@ export const BlockConcept = _ref4 => {
354
366
  audio: visualElement?.status === 'success' && visualElement.resource === 'audio' ? {
355
367
  src: visualElement.data.audioFile.url,
356
368
  title: visualElement.data.title.title
357
- } : undefined
369
+ } : undefined,
370
+ exampleIds: exampleIds,
371
+ exampleLangs: exampleLangs
358
372
  }), copyright && conceptType === 'concept' && _jsx(EmbedByline, {
359
373
  copyright: copyright,
360
374
  bottomRounded: true,