@ndla/ui 46.1.0 → 47.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/es/Article/Article.js +3 -17
- package/es/Embed/AudioEmbed.js +6 -14
- package/es/Embed/BrightcoveEmbed.js +13 -5
- package/es/Embed/ConceptEmbed.js +16 -14
- package/es/Embed/ImageEmbed.js +11 -4
- package/es/Embed/conceptComponents.js +11 -12
- package/es/Footer/FooterLinks.js +6 -6
- package/es/LicenseByline/EmbedByline.js +7 -15
- package/es/Messages/MessageBanner.js +4 -13
- package/es/Messages/MessageBox.js +10 -20
- package/es/Navigation/NavigationTopicAbout.js +15 -17
- package/es/Notion/Notion.js +8 -7
- package/es/SearchTypeResult/SearchFieldHeader.js +4 -4
- package/es/Topic/Topic.js +18 -20
- package/lib/Article/Article.d.ts +2 -4
- package/lib/Article/Article.js +3 -17
- package/lib/Embed/AudioEmbed.js +6 -15
- package/lib/Embed/BrightcoveEmbed.js +12 -4
- package/lib/Embed/ConceptEmbed.js +15 -13
- package/lib/Embed/ImageEmbed.js +10 -3
- package/lib/Embed/conceptComponents.d.ts +1 -1
- package/lib/Embed/conceptComponents.js +11 -12
- package/lib/Footer/FooterLinks.js +6 -6
- package/lib/LicenseByline/EmbedByline.d.ts +2 -2
- package/lib/LicenseByline/EmbedByline.js +8 -16
- package/lib/Messages/MessageBanner.d.ts +2 -1
- package/lib/Messages/MessageBanner.js +5 -13
- package/lib/Messages/MessageBox.d.ts +2 -1
- package/lib/Messages/MessageBox.js +11 -19
- package/lib/Navigation/NavigationTopicAbout.d.ts +2 -3
- package/lib/Navigation/NavigationTopicAbout.js +15 -17
- package/lib/Notion/Notion.js +7 -6
- package/lib/SearchTypeResult/SearchFieldHeader.js +4 -4
- package/lib/TagSelector/ariaMessages.d.ts +1 -1
- package/lib/Topic/Topic.d.ts +2 -3
- package/lib/Topic/Topic.js +18 -20
- package/lib/types.d.ts +1 -1
- package/package.json +19 -20
- package/src/Article/Article.tsx +2 -21
- package/src/Embed/AudioEmbed.tsx +5 -11
- package/src/Embed/BrightcoveEmbed.tsx +10 -7
- package/src/Embed/ConceptEmbed.stories.tsx +1 -1
- package/src/Embed/ConceptEmbed.tsx +9 -4
- package/src/Embed/ImageEmbed.tsx +12 -2
- package/src/Embed/conceptComponents.tsx +2 -3
- package/src/Footer/FooterLinks.tsx +1 -1
- package/src/LicenseByline/EmbedByline.tsx +4 -11
- package/src/Messages/MessageBanner.tsx +3 -8
- package/src/Messages/MessageBox.tsx +3 -8
- package/src/Navigation/NavigationTopicAbout.tsx +2 -5
- package/src/Notion/Notion.tsx +2 -2
- package/src/SearchTypeResult/SearchFieldHeader.tsx +4 -4
- package/src/TagSelector/ariaMessages.ts +1 -1
- package/src/Topic/Topic.tsx +2 -5
- package/src/types.ts +1 -1
package/es/Embed/ConceptEmbed.js
CHANGED
|
@@ -14,7 +14,8 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
|
|
|
14
14
|
*
|
|
15
15
|
*/
|
|
16
16
|
|
|
17
|
-
import { useCallback, useRef, useState } from 'react';
|
|
17
|
+
import { useCallback, useMemo, useRef, useState } from 'react';
|
|
18
|
+
import parse from 'html-react-parser';
|
|
18
19
|
import { useTranslation } from 'react-i18next';
|
|
19
20
|
import { isMobile } from 'react-device-detect';
|
|
20
21
|
import { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';
|
|
@@ -41,21 +42,25 @@ var PopoverWrapper = /*#__PURE__*/_styled("div", {
|
|
|
41
42
|
return top;
|
|
42
43
|
}, "px!important;}", mq.range({
|
|
43
44
|
until: breakpoints.tablet
|
|
44
|
-
}), "{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":"AAgCkD","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, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
45
|
+
}), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:100!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAiCkD","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\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  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
45
46
|
var ImageWrapper = /*#__PURE__*/_styled("div", {
|
|
46
47
|
target: "e6acljj5",
|
|
47
48
|
label: "ImageWrapper"
|
|
48
49
|
})("float:right;padding-left:", spacing.normal, ";position:relative;", mq.range({
|
|
49
50
|
until: breakpoints.tabletWide
|
|
50
|
-
}), "{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":"AAyD+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, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
51
|
+
}), "{width:100%;padding-left:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA0D+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\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  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
51
52
|
var StyledButton = /*#__PURE__*/_styled("button", {
|
|
52
53
|
target: "e6acljj4",
|
|
53
54
|
label: "StyledButton"
|
|
54
|
-
})("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":"AA0EkC","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, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
55
|
+
})("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":"AA2EkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\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  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
55
56
|
export var ConceptEmbed = function ConceptEmbed(_ref2) {
|
|
56
57
|
var embed = _ref2.embed,
|
|
57
58
|
fullWidth = _ref2.fullWidth,
|
|
58
59
|
HeartButton = _ref2.heartButton;
|
|
60
|
+
var parsedContent = useMemo(function () {
|
|
61
|
+
if (embed.status === 'error' || !embed.data.concept.content) return undefined;
|
|
62
|
+
return parse(embed.data.concept.content.content);
|
|
63
|
+
}, [embed]);
|
|
59
64
|
if (embed.status === 'error' && embed.embedData.type === 'inline') {
|
|
60
65
|
return _jsx("span", {
|
|
61
66
|
children: embed.embedData.linkText
|
|
@@ -69,11 +74,10 @@ export var ConceptEmbed = function ConceptEmbed(_ref2) {
|
|
|
69
74
|
concept = _embed$data.concept,
|
|
70
75
|
visualElement = _embed$data.visualElement;
|
|
71
76
|
if (embed.embedData.type === 'block') {
|
|
72
|
-
var _concept$content;
|
|
73
77
|
return _jsx(BlockConcept, {
|
|
74
78
|
fullWidth: fullWidth,
|
|
75
79
|
title: concept.title,
|
|
76
|
-
content:
|
|
80
|
+
content: parsedContent,
|
|
77
81
|
metaImage: concept.metaImage,
|
|
78
82
|
copyright: concept.copyright,
|
|
79
83
|
source: concept.source,
|
|
@@ -86,10 +90,9 @@ export var ConceptEmbed = function ConceptEmbed(_ref2) {
|
|
|
86
90
|
glossData: concept.glossData
|
|
87
91
|
});
|
|
88
92
|
} else if (embed.embedData.type === 'inline') {
|
|
89
|
-
var _concept$content2;
|
|
90
93
|
return _jsx(InlineConcept, {
|
|
91
94
|
title: concept.title,
|
|
92
|
-
content:
|
|
95
|
+
content: parsedContent,
|
|
93
96
|
metaImage: concept.metaImage,
|
|
94
97
|
copyright: concept.copyright,
|
|
95
98
|
source: concept.source,
|
|
@@ -103,10 +106,9 @@ export var ConceptEmbed = function ConceptEmbed(_ref2) {
|
|
|
103
106
|
glossData: concept.glossData
|
|
104
107
|
});
|
|
105
108
|
} else {
|
|
106
|
-
var _concept$content3;
|
|
107
109
|
return _jsx(ConceptNotionV2, {
|
|
108
110
|
title: concept.title,
|
|
109
|
-
content:
|
|
111
|
+
content: parsedContent,
|
|
110
112
|
metaImage: concept.metaImage,
|
|
111
113
|
copyright: concept.copyright,
|
|
112
114
|
source: concept.source,
|
|
@@ -129,19 +131,19 @@ var BaselineIcon = /*#__PURE__*/_styled("span", {
|
|
|
129
131
|
} : {
|
|
130
132
|
name: "6mki3j",
|
|
131
133
|
styles: "display:block;border-bottom:5px double currentColor",
|
|
132
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAiKgC","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, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
134
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAsKgC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\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  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
133
135
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
134
136
|
});
|
|
135
137
|
var NotionButton = /*#__PURE__*/_styled("button", {
|
|
136
138
|
target: "e6acljj2",
|
|
137
139
|
label: "NotionButton"
|
|
138
|
-
})("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":"AAsKkC","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, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
140
|
+
})("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":"AA2KkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\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  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
139
141
|
var StyledAnchor = /*#__PURE__*/_styled(Anchor, {
|
|
140
142
|
target: "e6acljj1",
|
|
141
143
|
label: "StyledAnchor"
|
|
142
144
|
})(mq.range({
|
|
143
145
|
until: breakpoints.tablet
|
|
144
|
-
}), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAuMmC","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, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
146
|
+
}), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA4MmC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\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  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
145
147
|
var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
|
|
146
148
|
target: "e6acljj0",
|
|
147
149
|
label: "StyledAnchorSpan"
|
|
@@ -151,7 +153,7 @@ var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
|
|
|
151
153
|
} : {
|
|
152
154
|
name: "zcodsq",
|
|
153
155
|
styles: "position:absolute;left:50%;align-self:center",
|
|
154
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA8MoC","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, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton }: Props) => {\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
156
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAmNoC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure } from '../Figure';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from './conceptComponents';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType } from './types';\nimport { Gloss } from '../Gloss';\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 100 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n}\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  const parsedContent = useMemo(() => {\n    if (embed.status === 'error' || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.content);\n  }, [embed]);\n  if (embed.status === 'error' && embed.embedData.type === 'inline') {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === 'error') {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest('.c-article') as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        {conceptType === 'concept' ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            visualElement={\n              visualElement?.status === 'success' && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                          {visualElement.resource === 'image' ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={''}\n                              src={metaImage?.url ?? ''}\n                              alt={metaImage?.alt ?? ''}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== 'undefined'\n                        ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === 'success' && visualElement.resource === 'audio'\n                ? { src: visualElement.data.audioFile.url, title: visualElement.data.title.title }\n                : undefined\n            }\n          />\n        )}\n        {copyright && conceptType === 'concept' && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
155
157
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
156
158
|
});
|
|
157
159
|
var getModalPosition = function getModalPosition(anchor) {
|