@ndla/ui 45.0.10 → 45.0.12

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.
@@ -36,7 +36,7 @@ import { Fragment as _Fragment } from "@emotion/react/jsx-runtime";
36
36
  var BottomBorder = /*#__PURE__*/_styled("div", {
37
37
  target: "e6acljj7",
38
38
  label: "BottomBorder"
39
- })("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":"AA4B+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
39
+ })("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":"AA4B+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
40
40
  var PopoverWrapper = /*#__PURE__*/_styled("div", {
41
41
  target: "e6acljj6",
42
42
  label: "PopoverWrapper"
@@ -45,17 +45,17 @@ var PopoverWrapper = /*#__PURE__*/_styled("div", {
45
45
  return top;
46
46
  }, "px!important;}", mq.range({
47
47
  until: breakpoints.tablet
48
- }), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:100!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAqCkD","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
48
+ }), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:100!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAqCkD","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
49
49
  var ImageWrapper = /*#__PURE__*/_styled("div", {
50
50
  target: "e6acljj5",
51
51
  label: "ImageWrapper"
52
52
  })("float:right;padding-left:", spacing.normal, ";position:relative;", mq.range({
53
53
  until: breakpoints.tabletWide
54
- }), "{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":"AA8D+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
54
+ }), "{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":"AA8D+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
55
55
  var StyledButton = /*#__PURE__*/_styled("button", {
56
56
  target: "e6acljj4",
57
57
  label: "StyledButton"
58
- })("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":"AA+EkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
58
+ })("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":"AA+EkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
59
59
  export var ConceptEmbed = function ConceptEmbed(_ref2) {
60
60
  var embed = _ref2.embed,
61
61
  fullWidth = _ref2.fullWidth,
@@ -133,19 +133,19 @@ var BaselineIcon = /*#__PURE__*/_styled("span", {
133
133
  } : {
134
134
  name: "6mki3j",
135
135
  styles: "display:block;border-bottom:5px double currentColor",
136
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAqKgC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
136
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAqKgC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
137
137
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
138
138
  });
139
139
  var NotionButton = /*#__PURE__*/_styled("button", {
140
140
  target: "e6acljj2",
141
141
  label: "NotionButton"
142
- })("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":"AA0KkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
142
+ })("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":"AA0KkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
143
143
  var StyledAnchor = /*#__PURE__*/_styled(Anchor, {
144
144
  target: "e6acljj1",
145
145
  label: "StyledAnchor"
146
146
  })(mq.range({
147
147
  until: breakpoints.tablet
148
- }), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA2MmC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
148
+ }), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA2MmC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
149
149
  var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
150
150
  target: "e6acljj0",
151
151
  label: "StyledAnchorSpan"
@@ -155,7 +155,7 @@ var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
155
155
  } : {
156
156
  name: "zcodsq",
157
157
  styles: "position:absolute;left:50%;align-self:center",
158
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAkNoC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
158
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAkNoC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, 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 { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { AudioMeta, ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\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: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.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        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData!}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
159
159
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
160
160
  });
161
161
  var getModalPosition = function getModalPosition(anchor) {
@@ -347,7 +347,7 @@ export var BlockConcept = function BlockConcept(_ref4) {
347
347
  src: visualElement.data.audioFile.url,
348
348
  title: visualElement.data.title.title
349
349
  } : undefined
350
- }), copyright ? _jsx(EmbedByline, {
350
+ }), copyright && conceptType === 'concept' ? _jsx(EmbedByline, {
351
351
  copyright: copyright,
352
352
  bottomRounded: true,
353
353
  topRounded: true,