@ndla/ui 34.6.2 → 34.6.3
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 +11 -6
- package/es/Aside/Aside.js +5 -2
- package/es/CopyParagraphButton/CopyParagraphButtonV2.js +85 -0
- package/es/CopyParagraphButton/index.js +2 -1
- package/es/Embed/AudioEmbed.js +254 -0
- package/es/Embed/BrightcoveEmbed.js +250 -0
- package/es/Embed/ConceptEmbed.js +358 -0
- package/es/Embed/ConceptListEmbed.js +71 -0
- package/es/Embed/ContentLinkEmbed.js +42 -0
- package/es/Embed/ExternalEmbed.js +91 -0
- package/es/Embed/FootnoteEmbed.js +32 -0
- package/es/Embed/H5pEmbed.js +87 -0
- package/es/Embed/IframeEmbed.js +83 -0
- package/es/Embed/ImageEmbed.js +322 -0
- package/es/Embed/RelatedContentEmbed.js +58 -0
- package/es/Embed/UnknownEmbed.js +27 -0
- package/es/Embed/conceptComponents.js +282 -0
- package/es/Embed/index.js +21 -0
- package/es/FactBox/FactBoxV2.js +90 -0
- package/es/FactBox/index.js +1 -0
- package/es/Figure/Figure.js +8 -5
- package/es/Figure/FigureLicenseDialogContent.js +72 -0
- package/es/FileList/FileListV2.js +47 -0
- package/es/FileList/FileV2.js +34 -0
- package/es/FileList/PdfFile.js +25 -0
- package/es/FileList/index.js +3 -0
- package/es/Notion/Notion.js +5 -5
- package/es/Notion/NotionVisualElement.js +2 -2
- package/es/RelatedArticleList/RelatedArticleV2.js +101 -0
- package/es/RelatedArticleList/index.js +2 -1
- package/es/Table/Table.js +95 -8
- package/es/all.css +1 -1
- package/es/index.js +5 -4
- package/es/locale/messages-en.js +8 -1
- package/es/locale/messages-nb.js +8 -1
- package/es/locale/messages-nn.js +8 -1
- package/es/locale/messages-se.js +8 -1
- package/es/locale/messages-sma.js +8 -1
- package/lib/Article/Article.d.ts +2 -1
- package/lib/Article/Article.js +11 -6
- package/lib/Aside/Aside.d.ts +2 -1
- package/lib/Aside/Aside.js +5 -2
- package/lib/CopyParagraphButton/CopyParagraphButtonV2.d.ts +14 -0
- package/lib/CopyParagraphButton/CopyParagraphButtonV2.js +84 -0
- package/lib/CopyParagraphButton/index.d.ts +2 -1
- package/lib/CopyParagraphButton/index.js +7 -0
- package/lib/Embed/AudioEmbed.d.ts +20 -0
- package/lib/Embed/AudioEmbed.js +252 -0
- package/lib/Embed/BrightcoveEmbed.d.ts +16 -0
- package/lib/Embed/BrightcoveEmbed.js +250 -0
- package/lib/Embed/ConceptEmbed.d.ts +19 -0
- package/lib/Embed/ConceptEmbed.js +358 -0
- package/lib/Embed/ConceptListEmbed.d.ts +13 -0
- package/lib/Embed/ConceptListEmbed.js +70 -0
- package/lib/Embed/ContentLinkEmbed.d.ts +14 -0
- package/lib/Embed/ContentLinkEmbed.js +50 -0
- package/lib/Embed/ExternalEmbed.d.ts +14 -0
- package/lib/Embed/ExternalEmbed.js +90 -0
- package/lib/Embed/FootnoteEmbed.d.ts +13 -0
- package/lib/Embed/FootnoteEmbed.js +39 -0
- package/lib/Embed/H5pEmbed.d.ts +14 -0
- package/lib/Embed/H5pEmbed.js +86 -0
- package/lib/Embed/IframeEmbed.d.ts +14 -0
- package/lib/Embed/IframeEmbed.js +91 -0
- package/lib/Embed/ImageEmbed.d.ts +37 -0
- package/lib/Embed/ImageEmbed.js +326 -0
- package/lib/Embed/RelatedContentEmbed.d.ts +16 -0
- package/lib/Embed/RelatedContentEmbed.js +64 -0
- package/lib/Embed/UnknownEmbed.d.ts +13 -0
- package/lib/Embed/UnknownEmbed.js +35 -0
- package/lib/Embed/conceptComponents.d.ts +32 -0
- package/lib/Embed/conceptComponents.js +280 -0
- package/lib/Embed/index.d.ts +20 -0
- package/lib/Embed/index.js +97 -0
- package/lib/FactBox/FactBoxV2.d.ts +13 -0
- package/lib/FactBox/FactBoxV2.js +92 -0
- package/lib/FactBox/index.d.ts +1 -0
- package/lib/FactBox/index.js +7 -0
- package/lib/Figure/Figure.d.ts +5 -2
- package/lib/Figure/Figure.js +8 -5
- package/lib/Figure/FigureLicenseDialogContent.d.ts +22 -0
- package/lib/Figure/FigureLicenseDialogContent.js +71 -0
- package/lib/FileList/FileListV2.d.ts +13 -0
- package/lib/FileList/FileListV2.js +46 -0
- package/lib/FileList/FileV2.d.ts +16 -0
- package/lib/FileList/FileV2.js +42 -0
- package/lib/FileList/PdfFile.d.ts +13 -0
- package/lib/FileList/PdfFile.js +31 -0
- package/lib/FileList/index.d.ts +3 -0
- package/lib/FileList/index.js +21 -0
- package/lib/Notion/Notion.js +5 -5
- package/lib/Notion/NotionVisualElement.d.ts +1 -1
- package/lib/Notion/NotionVisualElement.js +2 -2
- package/lib/RelatedArticleList/RelatedArticleV2.d.ts +25 -0
- package/lib/RelatedArticleList/RelatedArticleV2.js +101 -0
- package/lib/RelatedArticleList/index.d.ts +2 -1
- package/lib/RelatedArticleList/index.js +7 -0
- package/lib/Table/Table.js +98 -8
- package/lib/all.css +1 -1
- package/lib/index.d.ts +5 -4
- package/lib/index.js +117 -2
- package/lib/locale/messages-en.d.ts +7 -0
- package/lib/locale/messages-en.js +8 -1
- package/lib/locale/messages-nb.d.ts +7 -0
- package/lib/locale/messages-nb.js +8 -1
- package/lib/locale/messages-nn.d.ts +7 -0
- package/lib/locale/messages-nn.js +8 -1
- package/lib/locale/messages-se.d.ts +7 -0
- package/lib/locale/messages-se.js +8 -1
- package/lib/locale/messages-sma.d.ts +7 -0
- package/lib/locale/messages-sma.js +8 -1
- package/lib/types.d.ts +1 -1
- package/package.json +16 -12
- package/src/Article/Article.tsx +8 -3
- package/src/Aside/Aside.tsx +9 -1
- package/src/Aside/component.aside.scss +3 -0
- package/src/CopyParagraphButton/CopyParagraphButtonV2.tsx +84 -0
- package/src/CopyParagraphButton/index.tsx +2 -1
- package/src/Embed/AudioEmbed.tsx +249 -0
- package/src/Embed/BrightcoveEmbed.tsx +203 -0
- package/src/Embed/ConceptEmbed.tsx +408 -0
- package/src/Embed/ConceptListEmbed.tsx +64 -0
- package/src/Embed/ContentLinkEmbed.tsx +41 -0
- package/src/Embed/ExternalEmbed.tsx +80 -0
- package/src/Embed/FootnoteEmbed.tsx +30 -0
- package/src/Embed/H5pEmbed.tsx +74 -0
- package/src/Embed/IframeEmbed.tsx +84 -0
- package/src/Embed/ImageEmbed.tsx +314 -0
- package/src/Embed/RelatedContentEmbed.tsx +62 -0
- package/src/Embed/UnknownEmbed.tsx +27 -0
- package/src/Embed/conceptComponents.tsx +393 -0
- package/src/Embed/index.ts +21 -0
- package/src/FactBox/FactBoxV2.tsx +56 -0
- package/src/FactBox/index.ts +2 -0
- package/src/Figure/Figure.tsx +28 -15
- package/src/Figure/FigureLicenseDialogContent.tsx +80 -0
- package/src/Figure/component.figure.scss +0 -1
- package/src/FileList/FileListV2.tsx +58 -0
- package/src/FileList/FileV2.tsx +35 -0
- package/src/FileList/PdfFile.tsx +25 -0
- package/src/FileList/index.ts +3 -0
- package/src/Notion/Notion.tsx +0 -1
- package/src/Notion/NotionVisualElement.tsx +1 -1
- package/src/RelatedArticleList/RelatedArticleV2.tsx +84 -0
- package/src/RelatedArticleList/index.ts +2 -1
- package/src/Table/Table.tsx +77 -4
- package/src/index.ts +19 -4
- package/src/locale/messages-en.ts +7 -0
- package/src/locale/messages-nb.ts +7 -0
- package/src/locale/messages-nn.ts +7 -0
- package/src/locale/messages-se.ts +7 -0
- package/src/locale/messages-sma.ts +7 -0
- package/src/types.ts +1 -1
|
@@ -0,0 +1,358 @@
|
|
|
1
|
+
import _styled from "@emotion/styled/base";
|
|
2
|
+
function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
|
|
3
|
+
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
|
|
4
|
+
function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
|
|
5
|
+
function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
|
|
6
|
+
function _iterableToArrayLimit(arr, i) { var _i = arr == null ? null : typeof Symbol !== "undefined" && arr[Symbol.iterator] || arr["@@iterator"]; if (_i == null) return; var _arr = []; var _n = true; var _d = false; var _s, _e; try { for (_i = _i.call(arr); !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; }
|
|
7
|
+
function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
|
|
8
|
+
function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
9
|
+
/**
|
|
10
|
+
* Copyright (c) 2023-present, NDLA.
|
|
11
|
+
*
|
|
12
|
+
* This source code is licensed under the GPLv3 license found in the
|
|
13
|
+
* LICENSE file in the root directory of this source tree.
|
|
14
|
+
*
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import { useCallback, useRef, useState } from 'react';
|
|
18
|
+
import { useTranslation } from 'react-i18next';
|
|
19
|
+
import { isMobile } from 'react-device-detect';
|
|
20
|
+
import { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';
|
|
21
|
+
import { ButtonV2, IconButtonV2 } from '@ndla/button';
|
|
22
|
+
import { Cross } from '@ndla/icons/action';
|
|
23
|
+
import { breakpoints, colors, mq, spacing } from '@ndla/core';
|
|
24
|
+
import { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';
|
|
25
|
+
import { ModalV2 } from '@ndla/modal';
|
|
26
|
+
import Tooltip from '@ndla/tooltip';
|
|
27
|
+
import { Notion as UINotion } from '../Notion';
|
|
28
|
+
import { Figure, FigureCaption } from '../Figure';
|
|
29
|
+
import { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';
|
|
30
|
+
import { NotionImage } from '../Notion/NotionImage';
|
|
31
|
+
import { ConceptNotionV2 } from './conceptComponents';
|
|
32
|
+
import { jsx as _jsx } from "@emotion/react/jsx-runtime";
|
|
33
|
+
import { jsxs as _jsxs } from "@emotion/react/jsx-runtime";
|
|
34
|
+
import { Fragment as _Fragment } from "@emotion/react/jsx-runtime";
|
|
35
|
+
var BottomBorder = /*#__PURE__*/_styled("div", {
|
|
36
|
+
target: "e6acljj7",
|
|
37
|
+
label: "BottomBorder"
|
|
38
|
+
})("margin-top:", spacing.normal, ";border-bottom:1px solid ", colors.brand.greyLight, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA0B+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
39
|
+
var PopoverWrapper = /*#__PURE__*/_styled("div", {
|
|
40
|
+
target: "e6acljj6",
|
|
41
|
+
label: "PopoverWrapper"
|
|
42
|
+
})("div[data-radix-popper-content-wrapper]{position:absolute!important;left:50%!important;transform:translateX(-50%)!important;top:", function (_ref) {
|
|
43
|
+
var top = _ref.top;
|
|
44
|
+
return top;
|
|
45
|
+
}, "px!important;}", mq.range({
|
|
46
|
+
until: breakpoints.tablet
|
|
47
|
+
}), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:9999!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAmCkD","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
48
|
+
var ImageWrapper = /*#__PURE__*/_styled("div", {
|
|
49
|
+
target: "e6acljj5",
|
|
50
|
+
label: "ImageWrapper"
|
|
51
|
+
})("float:right;padding-left:", spacing.normal, ";position:relative;", mq.range({
|
|
52
|
+
until: breakpoints.tabletWide
|
|
53
|
+
}), "{width:100%;padding-left:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA4D+B","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
54
|
+
var StyledButton = /*#__PURE__*/_styled("button", {
|
|
55
|
+
target: "e6acljj4",
|
|
56
|
+
label: "StyledButton"
|
|
57
|
+
})("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;color:#000;position:relative;cursor:pointer;&:focus,&:hover{color:", colors.brand.primary, ";outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA4EkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
58
|
+
export var ConceptEmbed = function ConceptEmbed(_ref2) {
|
|
59
|
+
var embed = _ref2.embed,
|
|
60
|
+
fullWidth = _ref2.fullWidth;
|
|
61
|
+
if (embed.status === 'error') {
|
|
62
|
+
return _jsx("span", {
|
|
63
|
+
children: embed.embedData.linkText
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
var _embed$data = embed.data,
|
|
67
|
+
concept = _embed$data.concept,
|
|
68
|
+
visualElement = _embed$data.visualElement;
|
|
69
|
+
if (embed.embedData.type === 'block') {
|
|
70
|
+
var _concept$content;
|
|
71
|
+
return _jsx(BlockConcept, {
|
|
72
|
+
fullWidth: fullWidth,
|
|
73
|
+
title: concept.title.title,
|
|
74
|
+
content: (_concept$content = concept.content) === null || _concept$content === void 0 ? void 0 : _concept$content.content,
|
|
75
|
+
metaImage: concept.metaImage,
|
|
76
|
+
copyright: concept.copyright,
|
|
77
|
+
source: concept.source,
|
|
78
|
+
visualElement: visualElement
|
|
79
|
+
});
|
|
80
|
+
} else if (embed.embedData.type === 'inline') {
|
|
81
|
+
var _concept$content2;
|
|
82
|
+
return _jsx(InlineConcept, {
|
|
83
|
+
title: concept.title.title,
|
|
84
|
+
content: (_concept$content2 = concept.content) === null || _concept$content2 === void 0 ? void 0 : _concept$content2.content,
|
|
85
|
+
metaImage: concept.metaImage,
|
|
86
|
+
copyright: concept.copyright,
|
|
87
|
+
source: concept.source,
|
|
88
|
+
visualElement: visualElement,
|
|
89
|
+
linkText: embed.embedData.linkText
|
|
90
|
+
});
|
|
91
|
+
} else {
|
|
92
|
+
var _concept$content3;
|
|
93
|
+
return _jsx(ConceptNotionV2, {
|
|
94
|
+
title: concept.title.title,
|
|
95
|
+
content: (_concept$content3 = concept.content) === null || _concept$content3 === void 0 ? void 0 : _concept$content3.content,
|
|
96
|
+
metaImage: concept.metaImage,
|
|
97
|
+
copyright: concept.copyright,
|
|
98
|
+
source: concept.source,
|
|
99
|
+
visualElement: visualElement
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
};
|
|
103
|
+
var BaselineIcon = /*#__PURE__*/_styled("span", {
|
|
104
|
+
target: "e6acljj3",
|
|
105
|
+
label: "BaselineIcon"
|
|
106
|
+
})(process.env.NODE_ENV === "production" ? {
|
|
107
|
+
name: "6mki3j",
|
|
108
|
+
styles: "display:block;border-bottom:5px double currentColor"
|
|
109
|
+
} : {
|
|
110
|
+
name: "6mki3j",
|
|
111
|
+
styles: "display:block;border-bottom:5px double currentColor",
|
|
112
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAkJgC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
113
|
+
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
114
|
+
});
|
|
115
|
+
var NotionButton = /*#__PURE__*/_styled("button", {
|
|
116
|
+
target: "e6acljj2",
|
|
117
|
+
label: "NotionButton"
|
|
118
|
+
})("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;position:relative;text-align:left;display:inline;color:", colors.notion.dark, ";cursor:pointer;&:focus,&:hover{background-color:", colors.notion.dark, ";color:", colors.white, ";outline:none;", BaselineIcon, "{border-color:transparent;}}&:active{color:", colors.notion.dark, ";background-color:", colors.notion.light, ";", BaselineIcon, "{border-color:currentColor;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAuJkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
119
|
+
var StyledAnchor = /*#__PURE__*/_styled(Anchor, {
|
|
120
|
+
target: "e6acljj1",
|
|
121
|
+
label: "StyledAnchor"
|
|
122
|
+
})(mq.range({
|
|
123
|
+
until: breakpoints.tablet
|
|
124
|
+
}), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAwLmC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
125
|
+
var StyledAnchorSpan = /*#__PURE__*/_styled("span", {
|
|
126
|
+
target: "e6acljj0",
|
|
127
|
+
label: "StyledAnchorSpan"
|
|
128
|
+
})(process.env.NODE_ENV === "production" ? {
|
|
129
|
+
name: "zcodsq",
|
|
130
|
+
styles: "position:absolute;left:50%;align-self:center"
|
|
131
|
+
} : {
|
|
132
|
+
name: "zcodsq",
|
|
133
|
+
styles: "position:absolute;left:50%;align-self:center",
|
|
134
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA+LoC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useCallback, useRef, useState } from 'react';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { isMobile } from 'react-device-detect';\nimport { Root, Trigger, Content, Anchor, Close, Portal } from '@radix-ui/react-popover';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { Cross } from '@ndla/icons/action';\nimport { breakpoints, colors, mq, spacing } from '@ndla/core';\nimport { getGroupedContributorDescriptionList, getLicenseByAbbreviation, getLicenseCredits } from '@ndla/licenses';\nimport { ModalV2 } from '@ndla/modal';\nimport { ConceptMetaData } from '@ndla/types-embed';\nimport Tooltip from '@ndla/tooltip';\nimport { Notion as UINotion } from '../Notion';\nimport { Figure, FigureCaption } from '../Figure';\nimport { FigureLicenseDialogContent } from '../Figure/FigureLicenseDialogContent';\nimport { NotionImage } from '../Notion/NotionImage';\nimport { ConceptNotionV2, ConceptNotionData } from './conceptComponents';\n\nconst BottomBorder = styled.div`\n  margin-top: ${spacing.normal};\n  border-bottom: 1px solid ${colors.brand.greyLight};\n`;\n\ninterface PopoverPosition {\n  top?: number;\n}\n\nconst PopoverWrapper = styled.div<PopoverPosition>`\n  div[data-radix-popper-content-wrapper] {\n    position: absolute !important;\n    left: 50% !important;\n    transform: translateX(-50%) !important;\n    top: ${({ top }) => top}px !important;\n  }\n\n  ${mq.range({ until: breakpoints.tablet })} {\n    div[data-radix-popper-content-wrapper] {\n      // Fix for popover positioning on mobile.\n      // If we modify all popovers we break license icons.\n      // https://github.com/radix-ui/primitives/issues/1839\n      position: fixed !important;\n      transform: none !important;\n      top: 0 !important;\n      left: 0 !important;\n      width: 100vw;\n      z-index: 9999 !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth }: Props) => {\n  if (embed.status === 'error') {\n    return <span>{embed.embedData.linkText}</span>;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === 'block') {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  } else if (embed.embedData.type === 'inline') {\n    return (\n      <InlineConcept\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title.title}\n        content={concept.content?.content}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: string;\n}\n\nconst BaselineIcon = styled.span`\n  display: block;\n  border-bottom: 5px double currentColor;\n`;\n\nconst NotionButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  display: inline;\n  color: ${colors.notion.dark};\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    background-color: ${colors.notion.dark};\n    color: ${colors.white};\n    outline: none;\n    ${BaselineIcon} {\n      border-color: transparent;\n    }\n  }\n\n  &:active {\n    color: ${colors.notion.dark};\n    background-color: ${colors.notion.light};\n    ${BaselineIcon} {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = document.querySelector('.c-article');\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY);\n};\n\nconst InlineConcept = ({ title, content, copyright, source, visualElement, linkText }: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton>\n          {linkText}\n          {<BaselineIcon />}\n        </NotionButton>\n      </Trigger>\n      <Portal\n        container={\n          typeof document !== 'undefined'\n            ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n            : undefined\n        }>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n}: ConceptProps) => {\n  const { t, i18n } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const [isOpen, setIsOpen] = useState(false);\n  const licenseCredits = getLicenseCredits(copyright);\n  const { creators, rightsholders, processors } = licenseCredits;\n  const authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;\n  const visualElementType =\n    visualElement?.embedData.resource === 'brightcove' ? 'video' : visualElement?.embedData.resource;\n\n  const groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map((item) => ({\n    name: item.description,\n    type: item.label,\n  }));\n  const license = copyright?.license && getLicenseByAbbreviation(copyright?.license?.license, i18n.language);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? 'full' : 'full-column'}>\n        <UINotion\n          id=\"\"\n          title={title}\n          text={content}\n          visualElement={\n            visualElement?.status === 'success' && (\n              <>\n                <ImageWrapper>\n                  <Tooltip tooltip={t('searchPage.resultType.showNotion')}>\n                    <Trigger asChild>\n                      <StyledButton type=\"button\" aria-label={t('concept.showDescription', { title: title })}>\n                        {visualElement.resource === 'image' ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={visualElement.data.imageUrl}\n                            alt={visualElement.data.alttext.alttext}\n                          />\n                        ) : metaImage ? (\n                          <NotionImage\n                            type={visualElementType}\n                            id={''}\n                            src={metaImage?.url ?? ''}\n                            alt={metaImage?.alt ?? ''}\n                          />\n                        ) : undefined}\n                      </StyledButton>\n                    </Trigger>\n                  </Tooltip>\n                </ImageWrapper>\n                <Portal\n                  container={\n                    typeof document !== 'undefined'\n                      ? (document.querySelector('.c-article') as HTMLElement | null) || undefined\n                      : undefined\n                  }>\n                  <PopoverWrapper top={modalPos}>\n                    <Content avoidCollisions={false} asChild side=\"bottom\">\n                      <ConceptNotionV2\n                        title={title}\n                        content={content}\n                        copyright={copyright}\n                        source={source}\n                        visualElement={visualElement}\n                        inPopover\n                        closeButton={\n                          <Close asChild>\n                            <IconButtonV2 aria-label={t('close')} variant=\"ghost\">\n                              <Cross />\n                            </IconButtonV2>\n                          </Close>\n                        }\n                      />\n                    </Content>\n                  </PopoverWrapper>\n                </Portal>\n              </>\n            )\n          }\n        />\n        {copyright?.license && license ? (\n          <FigureCaption\n            figureId=\"\"\n            id=\"\"\n            authors={authors}\n            licenseRights={license.rights}\n            locale={i18n.language}\n            hideIconsAndAuthors\n            modalButton={\n              <ButtonV2 variant=\"outline\" size=\"small\" shape=\"pill\" onClick={() => setIsOpen(true)}>\n                {t('concept.reuse')}\n              </ButtonV2>\n            }>\n            <ModalV2\n              controlled\n              isOpen={isOpen}\n              onClose={() => setIsOpen(false)}\n              labelledBy=\"license-dialog-rules-heading\">\n              {(close) => (\n                <FigureLicenseDialogContent\n                  authors={groupedAuthors}\n                  locale={i18n.language}\n                  title={title}\n                  origin={copyright.origin}\n                  license={license}\n                  onClose={close}\n                  type=\"concept\"\n                />\n              )}\n            </ModalV2>\n          </FigureCaption>\n        ) : (\n          <BottomBorder />\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
135
|
+
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
136
|
+
});
|
|
137
|
+
var getModalPosition = function getModalPosition(anchor) {
|
|
138
|
+
var article = document.querySelector('.c-article');
|
|
139
|
+
var articlePos = article === null || article === void 0 ? void 0 : article.getBoundingClientRect();
|
|
140
|
+
var anchorPos = anchor.getBoundingClientRect();
|
|
141
|
+
return anchorPos.top - ((articlePos === null || articlePos === void 0 ? void 0 : articlePos.top) || -window.scrollY);
|
|
142
|
+
};
|
|
143
|
+
var InlineConcept = function InlineConcept(_ref3) {
|
|
144
|
+
var title = _ref3.title,
|
|
145
|
+
content = _ref3.content,
|
|
146
|
+
copyright = _ref3.copyright,
|
|
147
|
+
source = _ref3.source,
|
|
148
|
+
visualElement = _ref3.visualElement,
|
|
149
|
+
linkText = _ref3.linkText;
|
|
150
|
+
var _useTranslation = useTranslation(),
|
|
151
|
+
t = _useTranslation.t;
|
|
152
|
+
var anchorRef = useRef(null);
|
|
153
|
+
var _useState = useState(-9999),
|
|
154
|
+
_useState2 = _slicedToArray(_useState, 2),
|
|
155
|
+
modalPos = _useState2[0],
|
|
156
|
+
setModalPos = _useState2[1];
|
|
157
|
+
var onOpenChange = useCallback(function (open) {
|
|
158
|
+
if (open) {
|
|
159
|
+
var anchor = anchorRef.current;
|
|
160
|
+
if (anchor) {
|
|
161
|
+
var top = getModalPosition(anchor);
|
|
162
|
+
setModalPos(top);
|
|
163
|
+
}
|
|
164
|
+
} else {
|
|
165
|
+
setModalPos(-9999);
|
|
166
|
+
}
|
|
167
|
+
}, []);
|
|
168
|
+
return _jsxs(Root, {
|
|
169
|
+
modal: isMobile,
|
|
170
|
+
onOpenChange: onOpenChange,
|
|
171
|
+
children: [_jsx(StyledAnchor, {
|
|
172
|
+
ref: anchorRef,
|
|
173
|
+
asChild: true,
|
|
174
|
+
children: _jsx(StyledAnchorSpan, {})
|
|
175
|
+
}), _jsx(Trigger, {
|
|
176
|
+
asChild: true,
|
|
177
|
+
children: _jsxs(NotionButton, {
|
|
178
|
+
children: [linkText, _jsx(BaselineIcon, {})]
|
|
179
|
+
})
|
|
180
|
+
}), _jsx(Portal, {
|
|
181
|
+
container: typeof document !== 'undefined' ? document.querySelector('.c-article') || undefined : undefined,
|
|
182
|
+
children: _jsx(PopoverWrapper, {
|
|
183
|
+
top: modalPos,
|
|
184
|
+
children: _jsx(Content, {
|
|
185
|
+
avoidCollisions: false,
|
|
186
|
+
side: "bottom",
|
|
187
|
+
asChild: true,
|
|
188
|
+
children: _jsx(ConceptNotionV2, {
|
|
189
|
+
title: title,
|
|
190
|
+
content: content,
|
|
191
|
+
copyright: copyright,
|
|
192
|
+
source: source,
|
|
193
|
+
visualElement: visualElement,
|
|
194
|
+
inPopover: true,
|
|
195
|
+
closeButton: _jsx(Close, {
|
|
196
|
+
asChild: true,
|
|
197
|
+
children: _jsx(IconButtonV2, {
|
|
198
|
+
"aria-label": t('close'),
|
|
199
|
+
variant: "ghost",
|
|
200
|
+
children: _jsx(Cross, {})
|
|
201
|
+
})
|
|
202
|
+
})
|
|
203
|
+
})
|
|
204
|
+
})
|
|
205
|
+
})
|
|
206
|
+
})]
|
|
207
|
+
});
|
|
208
|
+
};
|
|
209
|
+
export var BlockConcept = function BlockConcept(_ref4) {
|
|
210
|
+
var _copyright$license, _metaImage$url, _metaImage$alt;
|
|
211
|
+
var title = _ref4.title,
|
|
212
|
+
content = _ref4.content,
|
|
213
|
+
metaImage = _ref4.metaImage,
|
|
214
|
+
copyright = _ref4.copyright,
|
|
215
|
+
source = _ref4.source,
|
|
216
|
+
visualElement = _ref4.visualElement,
|
|
217
|
+
fullWidth = _ref4.fullWidth;
|
|
218
|
+
var _useTranslation2 = useTranslation(),
|
|
219
|
+
t = _useTranslation2.t,
|
|
220
|
+
i18n = _useTranslation2.i18n;
|
|
221
|
+
var anchorRef = useRef(null);
|
|
222
|
+
var _useState3 = useState(-9999),
|
|
223
|
+
_useState4 = _slicedToArray(_useState3, 2),
|
|
224
|
+
modalPos = _useState4[0],
|
|
225
|
+
setModalPos = _useState4[1];
|
|
226
|
+
var _useState5 = useState(false),
|
|
227
|
+
_useState6 = _slicedToArray(_useState5, 2),
|
|
228
|
+
isOpen = _useState6[0],
|
|
229
|
+
setIsOpen = _useState6[1];
|
|
230
|
+
var licenseCredits = getLicenseCredits(copyright);
|
|
231
|
+
var creators = licenseCredits.creators,
|
|
232
|
+
rightsholders = licenseCredits.rightsholders,
|
|
233
|
+
processors = licenseCredits.processors;
|
|
234
|
+
var authors = creators.length || rightsholders.length ? creators.concat(rightsholders) : processors;
|
|
235
|
+
var visualElementType = (visualElement === null || visualElement === void 0 ? void 0 : visualElement.embedData.resource) === 'brightcove' ? 'video' : visualElement === null || visualElement === void 0 ? void 0 : visualElement.embedData.resource;
|
|
236
|
+
var groupedAuthors = getGroupedContributorDescriptionList(licenseCredits, i18n.language).map(function (item) {
|
|
237
|
+
return {
|
|
238
|
+
name: item.description,
|
|
239
|
+
type: item.label
|
|
240
|
+
};
|
|
241
|
+
});
|
|
242
|
+
var license = (copyright === null || copyright === void 0 ? void 0 : copyright.license) && getLicenseByAbbreviation(copyright === null || copyright === void 0 ? void 0 : (_copyright$license = copyright.license) === null || _copyright$license === void 0 ? void 0 : _copyright$license.license, i18n.language);
|
|
243
|
+
var onOpenChange = useCallback(function (open) {
|
|
244
|
+
if (open) {
|
|
245
|
+
var anchor = anchorRef.current;
|
|
246
|
+
if (anchor) {
|
|
247
|
+
var top = getModalPosition(anchor);
|
|
248
|
+
setModalPos(top);
|
|
249
|
+
}
|
|
250
|
+
} else {
|
|
251
|
+
setModalPos(-9999);
|
|
252
|
+
}
|
|
253
|
+
}, []);
|
|
254
|
+
return _jsxs(Root, {
|
|
255
|
+
modal: isMobile,
|
|
256
|
+
onOpenChange: onOpenChange,
|
|
257
|
+
children: [_jsx(StyledAnchor, {
|
|
258
|
+
ref: anchorRef
|
|
259
|
+
}), _jsxs(Figure, {
|
|
260
|
+
resizeIframe: true,
|
|
261
|
+
type: fullWidth ? 'full' : 'full-column',
|
|
262
|
+
children: [_jsx(UINotion, {
|
|
263
|
+
id: "",
|
|
264
|
+
title: title,
|
|
265
|
+
text: content,
|
|
266
|
+
visualElement: (visualElement === null || visualElement === void 0 ? void 0 : visualElement.status) === 'success' && _jsxs(_Fragment, {
|
|
267
|
+
children: [_jsx(ImageWrapper, {
|
|
268
|
+
children: _jsx(Tooltip, {
|
|
269
|
+
tooltip: t('searchPage.resultType.showNotion'),
|
|
270
|
+
children: _jsx(Trigger, {
|
|
271
|
+
asChild: true,
|
|
272
|
+
children: _jsx(StyledButton, {
|
|
273
|
+
type: "button",
|
|
274
|
+
"aria-label": t('concept.showDescription', {
|
|
275
|
+
title: title
|
|
276
|
+
}),
|
|
277
|
+
children: visualElement.resource === 'image' ? _jsx(NotionImage, {
|
|
278
|
+
type: visualElementType,
|
|
279
|
+
id: '',
|
|
280
|
+
src: visualElement.data.imageUrl,
|
|
281
|
+
alt: visualElement.data.alttext.alttext
|
|
282
|
+
}) : metaImage ? _jsx(NotionImage, {
|
|
283
|
+
type: visualElementType,
|
|
284
|
+
id: '',
|
|
285
|
+
src: (_metaImage$url = metaImage === null || metaImage === void 0 ? void 0 : metaImage.url) !== null && _metaImage$url !== void 0 ? _metaImage$url : '',
|
|
286
|
+
alt: (_metaImage$alt = metaImage === null || metaImage === void 0 ? void 0 : metaImage.alt) !== null && _metaImage$alt !== void 0 ? _metaImage$alt : ''
|
|
287
|
+
}) : undefined
|
|
288
|
+
})
|
|
289
|
+
})
|
|
290
|
+
})
|
|
291
|
+
}), _jsx(Portal, {
|
|
292
|
+
container: typeof document !== 'undefined' ? document.querySelector('.c-article') || undefined : undefined,
|
|
293
|
+
children: _jsx(PopoverWrapper, {
|
|
294
|
+
top: modalPos,
|
|
295
|
+
children: _jsx(Content, {
|
|
296
|
+
avoidCollisions: false,
|
|
297
|
+
asChild: true,
|
|
298
|
+
side: "bottom",
|
|
299
|
+
children: _jsx(ConceptNotionV2, {
|
|
300
|
+
title: title,
|
|
301
|
+
content: content,
|
|
302
|
+
copyright: copyright,
|
|
303
|
+
source: source,
|
|
304
|
+
visualElement: visualElement,
|
|
305
|
+
inPopover: true,
|
|
306
|
+
closeButton: _jsx(Close, {
|
|
307
|
+
asChild: true,
|
|
308
|
+
children: _jsx(IconButtonV2, {
|
|
309
|
+
"aria-label": t('close'),
|
|
310
|
+
variant: "ghost",
|
|
311
|
+
children: _jsx(Cross, {})
|
|
312
|
+
})
|
|
313
|
+
})
|
|
314
|
+
})
|
|
315
|
+
})
|
|
316
|
+
})
|
|
317
|
+
})]
|
|
318
|
+
})
|
|
319
|
+
}), copyright !== null && copyright !== void 0 && copyright.license && license ? _jsx(FigureCaption, {
|
|
320
|
+
figureId: "",
|
|
321
|
+
id: "",
|
|
322
|
+
authors: authors,
|
|
323
|
+
licenseRights: license.rights,
|
|
324
|
+
locale: i18n.language,
|
|
325
|
+
hideIconsAndAuthors: true,
|
|
326
|
+
modalButton: _jsx(ButtonV2, {
|
|
327
|
+
variant: "outline",
|
|
328
|
+
size: "small",
|
|
329
|
+
shape: "pill",
|
|
330
|
+
onClick: function onClick() {
|
|
331
|
+
return setIsOpen(true);
|
|
332
|
+
},
|
|
333
|
+
children: t('concept.reuse')
|
|
334
|
+
}),
|
|
335
|
+
children: _jsx(ModalV2, {
|
|
336
|
+
controlled: true,
|
|
337
|
+
isOpen: isOpen,
|
|
338
|
+
onClose: function onClose() {
|
|
339
|
+
return setIsOpen(false);
|
|
340
|
+
},
|
|
341
|
+
labelledBy: "license-dialog-rules-heading",
|
|
342
|
+
children: function children(close) {
|
|
343
|
+
return _jsx(FigureLicenseDialogContent, {
|
|
344
|
+
authors: groupedAuthors,
|
|
345
|
+
locale: i18n.language,
|
|
346
|
+
title: title,
|
|
347
|
+
origin: copyright.origin,
|
|
348
|
+
license: license,
|
|
349
|
+
onClose: close,
|
|
350
|
+
type: "concept"
|
|
351
|
+
});
|
|
352
|
+
}
|
|
353
|
+
})
|
|
354
|
+
}) : _jsx(BottomBorder, {})]
|
|
355
|
+
})]
|
|
356
|
+
});
|
|
357
|
+
};
|
|
358
|
+
export default ConceptEmbed;
|