@ndla/ui 45.0.0 → 45.0.2
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.
- package/es/Embed/ConceptEmbed.js +8 -8
- package/es/Embed/conceptComponents.js +6 -6
- package/es/Topic/Topic.js +17 -20
- package/lib/Embed/ConceptEmbed.js +8 -8
- package/lib/Embed/conceptComponents.js +6 -6
- package/lib/Topic/Topic.js +17 -20
- package/package.json +5 -5
- package/src/Embed/ConceptEmbed.tsx +1 -1
- package/src/Embed/conceptComponents.tsx +1 -1
- package/src/Topic/Topic.tsx +2 -7
package/es/Embed/ConceptEmbed.js
CHANGED
|
@@ -35,7 +35,7 @@ import { Fragment as _Fragment } from "@emotion/react/jsx-runtime";
|
|
|
35
35
|
var BottomBorder = /*#__PURE__*/_styled("div", {
|
|
36
36
|
target: "e6acljj7",
|
|
37
37
|
label: "BottomBorder"
|
|
38
|
-
})("margin-top:", spacing.normal, ";border-bottom:1px solid ", colors.brand.greyLight, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA2B+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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
38
|
+
})("margin-top:", spacing.normal, ";border-bottom:1px solid ", colors.brand.greyLight, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA2B+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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\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
39
|
var PopoverWrapper = /*#__PURE__*/_styled("div", {
|
|
40
40
|
target: "e6acljj6",
|
|
41
41
|
label: "PopoverWrapper"
|
|
@@ -44,17 +44,17 @@ var PopoverWrapper = /*#__PURE__*/_styled("div", {
|
|
|
44
44
|
return top;
|
|
45
45
|
}, "px!important;}", mq.range({
|
|
46
46
|
until: breakpoints.tablet
|
|
47
|
-
}), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:9999!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAoCkD","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
47
|
+
}), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index: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":"AAoCkD","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\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
48
|
var ImageWrapper = /*#__PURE__*/_styled("div", {
|
|
49
49
|
target: "e6acljj5",
|
|
50
50
|
label: "ImageWrapper"
|
|
51
51
|
})("float:right;padding-left:", spacing.normal, ";position:relative;", mq.range({
|
|
52
52
|
until: breakpoints.tabletWide
|
|
53
|
-
}), "{width:100%;padding-left:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA6D+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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
53
|
+
}), "{width:100%;padding-left:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA6D+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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\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
54
|
var StyledButton = /*#__PURE__*/_styled("button", {
|
|
55
55
|
target: "e6acljj4",
|
|
56
56
|
label: "StyledButton"
|
|
57
|
-
})("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;color:#000;position:relative;cursor:pointer;&:focus,&:hover{color:", colors.brand.primary, ";outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA8EkC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
57
|
+
})("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;color:#000;position:relative;cursor:pointer;&:focus,&:hover{color:", colors.brand.primary, ";outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA8EkC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\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
58
|
export var ConceptEmbed = function ConceptEmbed(_ref2) {
|
|
59
59
|
var embed = _ref2.embed,
|
|
60
60
|
fullWidth = _ref2.fullWidth,
|
|
@@ -126,19 +126,19 @@ var BaselineIcon = /*#__PURE__*/_styled("span", {
|
|
|
126
126
|
} : {
|
|
127
127
|
name: "6mki3j",
|
|
128
128
|
styles: "display:block;border-bottom:5px double currentColor",
|
|
129
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA8JgC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
129
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA8JgC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
130
130
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
131
131
|
});
|
|
132
132
|
var NotionButton = /*#__PURE__*/_styled("button", {
|
|
133
133
|
target: "e6acljj2",
|
|
134
134
|
label: "NotionButton"
|
|
135
|
-
})("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":"AAmKkC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
135
|
+
})("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":"AAmKkC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\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
136
|
var StyledAnchor = /*#__PURE__*/_styled(Anchor, {
|
|
137
137
|
target: "e6acljj1",
|
|
138
138
|
label: "StyledAnchor"
|
|
139
139
|
})(mq.range({
|
|
140
140
|
until: breakpoints.tablet
|
|
141
|
-
}), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAoMmC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
141
|
+
}), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAoMmC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\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
142
|
var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
|
|
143
143
|
target: "e6acljj0",
|
|
144
144
|
label: "StyledAnchorSpan"
|
|
@@ -148,7 +148,7 @@ var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
|
|
|
148
148
|
} : {
|
|
149
149
|
name: "zcodsq",
|
|
150
150
|
styles: "position:absolute;left:50%;align-self:center",
|
|
151
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA2MoC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
151
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA2MoC","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 { 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 } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\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.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      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\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}: 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            />\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}: 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        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.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                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright ? (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type=\"concept\">\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
152
152
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
153
153
|
});
|
|
154
154
|
var getModalPosition = function getModalPosition(anchor) {
|