@ndla/ui 50.12.0 → 50.12.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/es/Embed/ConceptEmbed.js +18 -27
- package/es/locale/messages-en.js +4 -7
- package/es/locale/messages-nb.js +4 -7
- package/es/locale/messages-nn.js +5 -8
- package/es/locale/messages-se.js +4 -7
- package/es/locale/messages-sma.js +4 -7
- package/lib/CompetenceGoalTab/styles.d.ts +5 -5
- package/lib/Embed/ConceptEmbed.js +18 -27
- package/lib/Hero/HeroContent.d.ts +1 -1
- package/lib/NDLAFilm/filmStyles.d.ts +2 -2
- package/lib/Search/ActiveFilterContent.d.ts +1 -1
- package/lib/Search/ContentTypeResultStyles.d.ts +6 -6
- package/lib/SearchTypeResult/ActiveFilterContent.d.ts +1 -1
- package/lib/TagSelector/MenuList.d.ts +1 -1
- package/lib/TagSelector/ValueButton.d.ts +1 -1
- package/lib/locale/messages-en.d.ts +1 -5
- package/lib/locale/messages-en.js +4 -10
- package/lib/locale/messages-nb.d.ts +1 -5
- package/lib/locale/messages-nb.js +4 -10
- package/lib/locale/messages-nn.d.ts +1 -5
- package/lib/locale/messages-nn.js +5 -11
- package/lib/locale/messages-se.d.ts +1 -5
- package/lib/locale/messages-se.js +4 -10
- package/lib/locale/messages-sma.d.ts +1 -5
- package/lib/locale/messages-sma.js +4 -10
- package/package.json +17 -17
- package/src/Embed/ConceptEmbed.tsx +7 -16
- package/src/LetterFilter/LetterFilter.stories.tsx +1 -1
- package/src/locale/messages-en.ts +2 -7
- package/src/locale/messages-nb.ts +3 -7
- package/src/locale/messages-nn.ts +3 -7
- package/src/locale/messages-se.ts +3 -7
- package/src/locale/messages-sma.ts +3 -7
package/es/Embed/ConceptEmbed.js
CHANGED
|
@@ -29,7 +29,7 @@ import { jsx as _jsx } from "@emotion/react/jsx-runtime";
|
|
|
29
29
|
import { jsxs as _jsxs } from "@emotion/react/jsx-runtime";
|
|
30
30
|
import { Fragment as _Fragment } from "@emotion/react/jsx-runtime";
|
|
31
31
|
const PopoverWrapper = /*#__PURE__*/_styled("div", {
|
|
32
|
-
target: "
|
|
32
|
+
target: "e6acljj5",
|
|
33
33
|
label: "PopoverWrapper"
|
|
34
34
|
})("div[data-radix-popper-content-wrapper]{position:absolute!important;left:50%!important;transform:translateX(-50%)!important;top:", _ref => {
|
|
35
35
|
let {
|
|
@@ -38,17 +38,17 @@ const PopoverWrapper = /*#__PURE__*/_styled("div", {
|
|
|
38
38
|
return top;
|
|
39
39
|
}, "px!important;z-index:", stackOrder.popover, "!important;}", mq.range({
|
|
40
40
|
until: breakpoints.tablet
|
|
41
|
-
}), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:", stackOrder.popover, "!important;height:100vh;min-width:100vw!important;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAiCkD","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: 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.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n    [data-baseline-icon] {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton data-open={modalPos !== -9999}>\n          {linkText}\n          {<BaselineIcon data-baseline-icon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
41
|
+
}), "{div[data-radix-popper-content-wrapper]{position:fixed!important;transform:none!important;top:0!important;left:0!important;width:100vw;z-index:", stackOrder.popover, "!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":"AAkCkD","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport { css } from \"@emotion/react\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst NotionButton = styled.span`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  color: ${colors.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n  }\n  display: inline;\n  border-bottom: 5px double currentColor;\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan contentEditable={false} />\n      </StyledAnchor>\n      <Trigger asChild type={undefined}>\n        <NotionButton role={\"button\"} data-open={modalPos !== -9999} tabIndex={0}>\n          {linkText}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
42
42
|
const ImageWrapper = /*#__PURE__*/_styled("div", {
|
|
43
|
-
target: "
|
|
43
|
+
target: "e6acljj4",
|
|
44
44
|
label: "ImageWrapper"
|
|
45
45
|
})("float:right;padding-left:", spacing.normal, ";position:relative;", mq.range({
|
|
46
46
|
until: breakpoints.tabletWide
|
|
47
|
-
}), "{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":"AA2D+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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: 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.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n    [data-baseline-icon] {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton data-open={modalPos !== -9999}>\n          {linkText}\n          {<BaselineIcon data-baseline-icon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
47
|
+
}), "{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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport { css } from \"@emotion/react\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst NotionButton = styled.span`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  color: ${colors.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n  }\n  display: inline;\n  border-bottom: 5px double currentColor;\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan contentEditable={false} />\n      </StyledAnchor>\n      <Trigger asChild type={undefined}>\n        <NotionButton role={\"button\"} data-open={modalPos !== -9999} tabIndex={0}>\n          {linkText}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
48
48
|
const StyledButton = /*#__PURE__*/_styled("button", {
|
|
49
|
-
target: "
|
|
49
|
+
target: "e6acljj3",
|
|
50
50
|
label: "StyledButton"
|
|
51
|
-
})("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":"AA6EkC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: 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.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n    [data-baseline-icon] {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton data-open={modalPos !== -9999}>\n          {linkText}\n          {<BaselineIcon data-baseline-icon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
51
|
+
})("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;padding:0 0 4px 0;margin-bottom:-4px;text-decoration:none;color:#000;position:relative;cursor:pointer;&:focus,&:hover{color:", colors.brand.primary, ";outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AA8EkC","file":"ConceptEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport { css } from \"@emotion/react\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst NotionButton = styled.span`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  color: ${colors.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n  }\n  display: inline;\n  border-bottom: 5px double currentColor;\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan contentEditable={false} />\n      </StyledAnchor>\n      <Trigger asChild type={undefined}>\n        <NotionButton role={\"button\"} data-open={modalPos !== -9999} tabIndex={0}>\n          {linkText}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
52
52
|
export const ConceptEmbed = _ref2 => {
|
|
53
53
|
let {
|
|
54
54
|
embed,
|
|
@@ -132,28 +132,16 @@ export const ConceptEmbed = _ref2 => {
|
|
|
132
132
|
});
|
|
133
133
|
}
|
|
134
134
|
};
|
|
135
|
-
const
|
|
136
|
-
target: "e6acljj3",
|
|
137
|
-
label: "BaselineIcon"
|
|
138
|
-
})(process.env.NODE_ENV === "production" ? {
|
|
139
|
-
name: "6mki3j",
|
|
140
|
-
styles: "display:block;border-bottom:5px double currentColor"
|
|
141
|
-
} : {
|
|
142
|
-
name: "6mki3j",
|
|
143
|
-
styles: "display:block;border-bottom:5px double currentColor",
|
|
144
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAkLgC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: 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.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n    [data-baseline-icon] {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton data-open={modalPos !== -9999}>\n          {linkText}\n          {<BaselineIcon data-baseline-icon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
145
|
-
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
146
|
-
});
|
|
147
|
-
const NotionButton = /*#__PURE__*/_styled("button", {
|
|
135
|
+
const NotionButton = /*#__PURE__*/_styled("span", {
|
|
148
136
|
target: "e6acljj2",
|
|
149
137
|
label: "NotionButton"
|
|
150
|
-
})("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.concept.text, ";cursor:pointer;&:focus,&:hover,&:active,&[data-open=\"true\"]{color:", colors.concept.text, ";background-color:", colors.concept.light, ";[data-baseline-icon]{border-color:currentColor;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAuLkC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: 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.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n    [data-baseline-icon] {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton data-open={modalPos !== -9999}>\n          {linkText}\n          {<BaselineIcon data-baseline-icon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
138
|
+
})("background:none;border:none;font-family:inherit;font-style:inherit;line-height:1em;text-decoration:none;position:relative;text-align:left;color:", colors.concept.text, ";cursor:pointer;&:focus,&:hover,&:active,&[data-open=\"true\"]{color:", colors.concept.text, ";background-color:", colors.concept.light, ";}display:inline;border-bottom:5px double currentColor;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAmLgC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport { css } from \"@emotion/react\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst NotionButton = styled.span`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  color: ${colors.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n  }\n  display: inline;\n  border-bottom: 5px double currentColor;\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan contentEditable={false} />\n      </StyledAnchor>\n      <Trigger asChild type={undefined}>\n        <NotionButton role={\"button\"} data-open={modalPos !== -9999} tabIndex={0}>\n          {linkText}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
151
139
|
const StyledAnchor = /*#__PURE__*/_styled(Anchor, {
|
|
152
140
|
target: "e6acljj1",
|
|
153
141
|
label: "StyledAnchor"
|
|
154
142
|
})(mq.range({
|
|
155
143
|
until: breakpoints.tablet
|
|
156
|
-
}), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAiNmC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: 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.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n    [data-baseline-icon] {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton data-open={modalPos !== -9999}>\n          {linkText}\n          {<BaselineIcon data-baseline-icon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
144
|
+
}), "{position:fixed;top:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAyMmC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport { css } from \"@emotion/react\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst NotionButton = styled.span`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  color: ${colors.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n  }\n  display: inline;\n  border-bottom: 5px double currentColor;\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan contentEditable={false} />\n      </StyledAnchor>\n      <Trigger asChild type={undefined}>\n        <NotionButton role={\"button\"} data-open={modalPos !== -9999} tabIndex={0}>\n          {linkText}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */"));
|
|
157
145
|
const StyledAnchorSpan = /*#__PURE__*/_styled("span", {
|
|
158
146
|
target: "e6acljj0",
|
|
159
147
|
label: "StyledAnchorSpan"
|
|
@@ -163,7 +151,7 @@ const StyledAnchorSpan = /*#__PURE__*/_styled("span", {
|
|
|
163
151
|
} : {
|
|
164
152
|
name: "zcodsq",
|
|
165
153
|
styles: "position:absolute;left:50%;align-self:center",
|
|
166
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAwNoC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: 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.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n    [data-baseline-icon] {\n      border-color: currentColor;\n    }\n  }\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan />\n      </StyledAnchor>\n      <Trigger asChild>\n        <NotionButton data-open={modalPos !== -9999}>\n          {linkText}\n          {<BaselineIcon data-baseline-icon />}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
154
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ConceptEmbed.tsx"],"names":[],"mappings":"AAgNoC","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 parse from \"html-react-parser\";\nimport { ReactElement, ReactNode, useCallback, useMemo, useRef, useState } from \"react\";\nimport { isMobile } from \"react-device-detect\";\nimport { useTranslation } from \"react-i18next\";\nimport { css } from \"@emotion/react\";\nimport styled from \"@emotion/styled\";\nimport { Root, Trigger, Content, Anchor, Close, Portal } from \"@radix-ui/react-popover\";\nimport { IconButtonV2 } from \"@ndla/button\";\nimport { breakpoints, colors, mq, spacing, stackOrder } from \"@ndla/core\";\nimport { Cross } from \"@ndla/icons/action\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { Tooltip } from \"@ndla/tooltip\";\nimport { ConceptMetaData } from \"@ndla/types-embed\";\nimport { ConceptNotionV2, ConceptNotionData, ConceptType } from \"./conceptComponents\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { HeartButtonType } from \"./types\";\nimport { Figure } from \"../Figure\";\nimport { Gloss } from \"../Gloss\";\nimport { EmbedByline } from \"../LicenseByline\";\nimport { Notion as UINotion } from \"../Notion\";\nimport { NotionImage } from \"../Notion/NotionImage\";\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    z-index: ${stackOrder.popover} !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: ${stackOrder.popover} !important;\n      height: 100vh;\n      min-width: 100vw !important;\n    }\n  }\n`;\n\nconst ImageWrapper = styled.div`\n  float: right;\n  padding-left: ${spacing.normal};\n  position: relative;\n\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    width: 100%;\n    padding-left: 0;\n  }\n`;\n\ninterface Props {\n  embed: ConceptMetaData;\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  lang?: string;\n}\n\nconst StyledButton = styled.button`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  padding: 0 0 4px 0;\n  margin-bottom: -4px;\n  text-decoration: none;\n  color: #000;\n  position: relative;\n  cursor: pointer;\n  &:focus,\n  &:hover {\n    color: ${colors.brand.primary};\n    outline: none;\n  }\n`;\n\nexport const ConceptEmbed = ({ embed, fullWidth, heartButton: HeartButton, lang }: Props) => {\n  const parsedContent = useMemo(() => {\n    if (embed.status === \"error\" || !embed.data.concept.content) return undefined;\n    return parse(embed.data.concept.content.htmlContent);\n  }, [embed]);\n  if (embed.status === \"error\" && embed.embedData.type === \"inline\") {\n    return <span>{embed.embedData.linkText}</span>;\n  } else if (embed.status === \"error\") {\n    return <EmbedErrorPlaceholder type=\"concept\" />;\n  }\n\n  const {\n    data: { concept, visualElement },\n  } = embed;\n\n  if (embed.embedData.type === \"block\") {\n    return (\n      <BlockConcept\n        fullWidth={fullWidth}\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else if (embed.embedData.type === \"inline\") {\n    return (\n      <InlineConcept\n        title={concept.title}\n        content={parsedContent}\n        metaImage={concept.metaImage}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        linkText={embed.embedData.linkText}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  } else {\n    return (\n      <ConceptNotionV2\n        title={concept.title}\n        content={parsedContent}\n        copyright={concept.copyright}\n        source={concept.source}\n        visualElement={visualElement}\n        heartButton={HeartButton}\n        conceptHeartButton={HeartButton && <HeartButton embed={embed} />}\n        conceptType={concept.conceptType}\n        glossData={concept.glossData}\n        lang={lang}\n        exampleIds={embed.embedData.exampleIds}\n        exampleLangs={embed.embedData.exampleLangs}\n      />\n    );\n  }\n};\n\ninterface InlineConceptProps extends ConceptNotionData {\n  linkText: ReactNode;\n  heartButton?: HeartButtonType;\n  headerButtons?: ReactNode;\n  conceptHeartButton?: ReactNode;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nconst NotionButton = styled.span`\n  background: none;\n  border: none;\n  font-family: inherit;\n  font-style: inherit;\n  line-height: 1em;\n  text-decoration: none;\n  position: relative;\n  text-align: left;\n  color: ${colors.concept.text};\n  cursor: pointer;\n  &:focus,\n  &:hover,\n  &:active,\n  &[data-open=\"true\"] {\n    color: ${colors.concept.text};\n    background-color: ${colors.concept.light};\n  }\n  display: inline;\n  border-bottom: 5px double currentColor;\n`;\n\nconst StyledAnchor = styled(Anchor)`\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    top: 0;\n  }\n`;\n\nconst StyledAnchorSpan = styled.span`\n  position: absolute;\n  left: 50%;\n  align-self: center;\n`;\n\nconst getModalPosition = (anchor: HTMLElement) => {\n  const article = anchor.closest(\".c-article\");\n  const articlePos = article?.getBoundingClientRect();\n  const anchorPos = anchor.getBoundingClientRect();\n  return anchorPos.top - (articlePos?.top || -window.scrollY) + 30; // add 30 so that position is under the word\n};\n\nexport const InlineConcept = ({\n  title,\n  content,\n  copyright,\n  source,\n  visualElement,\n  linkText,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  headerButtons,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: InlineConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} asChild>\n        <StyledAnchorSpan contentEditable={false} />\n      </StyledAnchor>\n      <Trigger asChild type={undefined}>\n        <NotionButton role={\"button\"} data-open={modalPos !== -9999} tabIndex={0}>\n          {linkText}\n        </NotionButton>\n      </Trigger>\n      <Portal container={(anchorRef.current?.closest(\".c-article\") as HTMLElement | null) || undefined}>\n        <PopoverWrapper top={modalPos}>\n          <Content avoidCollisions={false} side=\"bottom\" asChild>\n            <ConceptNotionV2\n              title={title}\n              content={content}\n              copyright={copyright}\n              source={source}\n              visualElement={visualElement}\n              inPopover\n              heartButton={heartButton}\n              headerButtons={headerButtons}\n              conceptHeartButton={conceptHeartButton}\n              lang={lang}\n              closeButton={\n                <Close asChild>\n                  <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                    <Cross />\n                  </IconButtonV2>\n                </Close>\n              }\n              conceptType={conceptType}\n              glossData={glossData}\n              exampleIds={exampleIds}\n              exampleLangs={exampleLangs}\n            />\n          </Content>\n        </PopoverWrapper>\n      </Portal>\n    </Root>\n  );\n};\n\ninterface ConceptProps extends ConceptNotionData {\n  fullWidth?: boolean;\n  heartButton?: HeartButtonType;\n  conceptHeartButton?: ReactElement;\n  exampleIds?: string;\n  exampleLangs?: string;\n}\n\nexport const BlockConcept = ({\n  title,\n  content,\n  metaImage,\n  copyright,\n  source,\n  visualElement,\n  fullWidth,\n  heartButton,\n  conceptHeartButton,\n  glossData,\n  conceptType,\n  lang,\n  exampleIds,\n  exampleLangs,\n}: ConceptProps) => {\n  const { t } = useTranslation();\n  const anchorRef = useRef<HTMLDivElement>(null);\n  const [modalPos, setModalPos] = useState(-9999);\n\n  const visualElementType =\n    visualElement?.embedData.resource === \"brightcove\" ? \"video\" : visualElement?.embedData.resource;\n\n  const onOpenChange = useCallback((open: boolean) => {\n    if (open) {\n      const anchor = anchorRef.current;\n      if (anchor) {\n        const top = getModalPosition(anchor);\n        setModalPos(top);\n      }\n    } else {\n      setModalPos(-9999);\n    }\n  }, []);\n\n  return (\n    <Root modal={isMobile} onOpenChange={onOpenChange}>\n      <StyledAnchor ref={anchorRef} />\n      <Figure resizeIframe type={fullWidth ? \"full\" : \"full-column\"}>\n        {conceptType === \"concept\" ? (\n          <UINotion\n            id=\"\"\n            title={title.title}\n            text={content}\n            lang={lang}\n            visualElement={\n              visualElement?.status === \"success\" && (\n                <>\n                  <ImageWrapper>\n                    <Tooltip tooltip={t(`searchPage.resultType.${conceptType}`)}>\n                      <Trigger asChild>\n                        <StyledButton\n                          type=\"button\"\n                          aria-label={t(\"concept.showDescription\", {\n                            title: title,\n                          })}\n                        >\n                          {visualElement.resource === \"image\" ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={visualElement.data.image.imageUrl}\n                              alt={visualElement.data.alttext.alttext}\n                            />\n                          ) : metaImage ? (\n                            <NotionImage\n                              type={visualElementType}\n                              id={\"\"}\n                              src={metaImage?.url ?? \"\"}\n                              alt={metaImage?.alt ?? \"\"}\n                            />\n                          ) : undefined}\n                        </StyledButton>\n                      </Trigger>\n                    </Tooltip>\n                  </ImageWrapper>\n                  <Portal\n                    container={\n                      typeof document !== \"undefined\"\n                        ? (document.querySelector(\".c-article\") as HTMLElement | null) || undefined\n                        : undefined\n                    }\n                  >\n                    <PopoverWrapper top={modalPos}>\n                      <Content avoidCollisions={false} asChild side=\"bottom\">\n                        <ConceptNotionV2\n                          title={title}\n                          content={content}\n                          copyright={copyright}\n                          source={source}\n                          visualElement={visualElement}\n                          heartButton={heartButton}\n                          conceptHeartButton={conceptHeartButton}\n                          inPopover\n                          lang={lang}\n                          closeButton={\n                            <Close asChild>\n                              <IconButtonV2 aria-label={t(\"close\")} variant=\"ghost\">\n                                <Cross />\n                              </IconButtonV2>\n                            </Close>\n                          }\n                          conceptType={conceptType}\n                          glossData={glossData}\n                        />\n                      </Content>\n                    </PopoverWrapper>\n                  </Portal>\n                </>\n              )\n            }\n          />\n        ) : (\n          <Gloss\n            glossData={glossData}\n            title={title}\n            audio={\n              visualElement?.status === \"success\" && visualElement.resource === \"audio\"\n                ? {\n                    src: visualElement.data.audioFile.url,\n                    title: visualElement.data.title.title,\n                  }\n                : undefined\n            }\n            exampleIds={exampleIds}\n            exampleLangs={exampleLangs}\n          />\n        )}\n        {copyright && conceptType === \"concept\" && (\n          <EmbedByline copyright={copyright} bottomRounded topRounded type={conceptType as ConceptType}>\n            {copyright.license?.license.toLowerCase() !== COPYRIGHTED && conceptHeartButton}\n          </EmbedByline>\n        )}\n      </Figure>\n    </Root>\n  );\n};\n\nexport default ConceptEmbed;\n"]} */",
|
|
167
155
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
168
156
|
});
|
|
169
157
|
const getModalPosition = anchor => {
|
|
@@ -211,14 +199,17 @@ export const InlineConcept = _ref3 => {
|
|
|
211
199
|
children: [_jsx(StyledAnchor, {
|
|
212
200
|
ref: anchorRef,
|
|
213
201
|
asChild: true,
|
|
214
|
-
children: _jsx(StyledAnchorSpan, {
|
|
202
|
+
children: _jsx(StyledAnchorSpan, {
|
|
203
|
+
contentEditable: false
|
|
204
|
+
})
|
|
215
205
|
}), _jsx(Trigger, {
|
|
216
206
|
asChild: true,
|
|
217
|
-
|
|
207
|
+
type: undefined,
|
|
208
|
+
children: _jsx(NotionButton, {
|
|
209
|
+
role: "button",
|
|
218
210
|
"data-open": modalPos !== -9999,
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
})]
|
|
211
|
+
tabIndex: 0,
|
|
212
|
+
children: linkText
|
|
222
213
|
})
|
|
223
214
|
}), _jsx(Portal, {
|
|
224
215
|
container: anchorRef.current?.closest(".c-article") || undefined,
|