@ndla/ui 34.6.3 → 34.6.4

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.
@@ -35,7 +35,7 @@ import { Fragment as _Fragment } from "@emotion/react/jsx-runtime";
35
35
  var BottomBorder = /*#__PURE__*/_styled("div", {
36
36
  target: "e6acljj7",
37
37
  label: "BottomBorder"
38
- })("margin-top:", spacing.normal, ";border-bottom:1px solid ", colors.brand.greyLight, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA0B+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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
38
+ })("margin-top:", spacing.normal, ";border-bottom:1px solid ", colors.brand.greyLight, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA0B+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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
39
39
  var PopoverWrapper = /*#__PURE__*/_styled("div", {
40
40
  target: "e6acljj6",
41
41
  label: "PopoverWrapper"
@@ -44,17 +44,17 @@ var PopoverWrapper = /*#__PURE__*/_styled("div", {
44
44
  return top;
45
45
  }, "px!important;}", mq.range({
46
46
  until: breakpoints.tablet
47
- }), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:9999!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":"AAmCkD","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
47
+ }), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:9999!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":"AAmCkD","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
48
48
  var ImageWrapper = /*#__PURE__*/_styled("div", {
49
49
  target: "e6acljj5",
50
50
  label: "ImageWrapper"
51
51
  })("float:right;padding-left:", spacing.normal, ";position:relative;", mq.range({
52
52
  until: breakpoints.tabletWide
53
- }), "{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":"AA4D+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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
53
+ }), "{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":"AA4D+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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
54
54
  var StyledButton = /*#__PURE__*/_styled("button", {
55
55
  target: "e6acljj4",
56
56
  label: "StyledButton"
57
- })("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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
57
+ })("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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
58
58
  export var ConceptEmbed = function ConceptEmbed(_ref2) {
59
59
  var embed = _ref2.embed,
60
60
  fullWidth = _ref2.fullWidth;
@@ -109,19 +109,19 @@ var BaselineIcon = /*#__PURE__*/_styled("span", {
109
109
  } : {
110
110
  name: "6mki3j",
111
111
  styles: "display:block;border-bottom:5px double currentColor",
112
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAkJgC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
112
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAkJgC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
113
113
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
114
114
  });
115
115
  var NotionButton = /*#__PURE__*/_styled("button", {
116
116
  target: "e6acljj2",
117
117
  label: "NotionButton"
118
- })("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":"AAuJkC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
118
+ })("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":"AAuJkC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
119
119
  var StyledAnchor = /*#__PURE__*/_styled(Anchor, {
120
120
  target: "e6acljj1",
121
121
  label: "StyledAnchor"
122
122
  })(mq.range({
123
123
  until: breakpoints.tablet
124
- }), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAwLmC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
124
+ }), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAwLmC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
125
125
  var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
126
126
  target: "e6acljj0",
127
127
  label: "StyledAnchorSpan"
@@ -131,16 +131,17 @@ var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
131
131
  } : {
132
132
  name: "zcodsq",
133
133
  styles: "position:absolute;left:50%;align-self:center",
134
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA+LoC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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 = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
134
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA+LoC","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 { useCallback, useRef, useState } from 'react';\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 { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\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: 9999 !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}\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 }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\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.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: 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\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: 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              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\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.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                  <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                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
135
135
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
136
136
  });
137
137
  var getModalPosition = function getModalPosition(anchor) {
138
- var article = document.querySelector('.c-article');
138
+ var article = anchor.closest('.c-article');
139
139
  var articlePos = article === null || article === void 0 ? void 0 : article.getBoundingClientRect();
140
140
  var anchorPos = anchor.getBoundingClientRect();
141
141
  return anchorPos.top - ((articlePos === null || articlePos === void 0 ? void 0 : articlePos.top) || -window.scrollY);
142
142
  };
143
143
  var InlineConcept = function InlineConcept(_ref3) {
144
+ var _anchorRef$current;
144
145
  var title = _ref3.title,
145
146
  content = _ref3.content,
146
147
  copyright = _ref3.copyright,
@@ -178,7 +179,7 @@ var InlineConcept = function InlineConcept(_ref3) {
178
179
  children: [linkText, _jsx(BaselineIcon, {})]
179
180
  })
180
181
  }), _jsx(Portal, {
181
- container: typeof document !== 'undefined' ? document.querySelector('.c-article') || undefined : undefined,
182
+ container: ((_anchorRef$current = anchorRef.current) === null || _anchorRef$current === void 0 ? void 0 : _anchorRef$current.closest('.c-article')) || undefined,
182
183
  children: _jsx(PopoverWrapper, {
183
184
  top: modalPos,
184
185
  children: _jsx(Content, {