@ndla/ui 50.15.3 → 51.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/es/Embed/ImageEmbed.js +4 -3
- package/es/Embed/index.js +1 -1
- package/es/Footer/FooterBlock.js +17 -0
- package/es/Footer/index.js +1 -4
- package/es/Subject/index.js +0 -1
- package/es/index.js +3 -3
- package/es/locale/messages-en.js +2 -1
- package/es/locale/messages-nb.js +2 -1
- package/es/locale/messages-nn.js +2 -1
- package/es/locale/messages-se.js +2 -1
- package/es/locale/messages-sma.js +2 -1
- package/lib/Embed/ImageEmbed.js +4 -3
- package/lib/Embed/index.d.ts +1 -1
- package/lib/Embed/index.js +15 -3
- package/lib/Footer/FooterBlock.d.ts +13 -0
- package/lib/Footer/FooterBlock.js +24 -0
- package/lib/Footer/index.d.ts +1 -4
- package/lib/Footer/index.js +3 -18
- package/lib/Subject/index.d.ts +0 -1
- package/lib/Subject/index.js +0 -7
- package/lib/index.d.ts +3 -3
- package/lib/index.js +14 -20
- package/lib/locale/messages-en.d.ts +1 -0
- package/lib/locale/messages-en.js +2 -1
- package/lib/locale/messages-nb.d.ts +1 -0
- package/lib/locale/messages-nb.js +2 -1
- package/lib/locale/messages-nn.d.ts +1 -0
- package/lib/locale/messages-nn.js +2 -1
- package/lib/locale/messages-se.d.ts +1 -0
- package/lib/locale/messages-se.js +2 -1
- package/lib/locale/messages-sma.d.ts +1 -0
- package/lib/locale/messages-sma.js +2 -1
- package/package.json +2 -2
- package/src/Embed/ImageEmbed.tsx +1 -0
- package/src/Embed/index.ts +1 -1
- package/src/Footer/Footer.stories.tsx +7 -105
- package/src/Footer/FooterBlock.tsx +30 -0
- package/src/Footer/index.ts +1 -5
- package/src/Subject/index.ts +0 -1
- package/src/index.ts +4 -2
- package/src/locale/messages-en.ts +1 -0
- package/src/locale/messages-nb.ts +1 -0
- package/src/locale/messages-nn.ts +1 -0
- package/src/locale/messages-se.ts +1 -0
- package/src/locale/messages-sma.ts +1 -0
- package/es/Footer/EditorName.js +0 -20
- package/es/Footer/Footer.js +0 -83
- package/es/Footer/FooterLinks.js +0 -108
- package/es/Footer/FooterText.js +0 -25
- package/es/Subject/SubjectHeader.js +0 -66
- package/lib/Footer/EditorName.d.ts +0 -13
- package/lib/Footer/EditorName.js +0 -27
- package/lib/Footer/Footer.d.ts +0 -30
- package/lib/Footer/Footer.js +0 -88
- package/lib/Footer/FooterLinks.d.ts +0 -26
- package/lib/Footer/FooterLinks.js +0 -115
- package/lib/Footer/FooterText.d.ts +0 -13
- package/lib/Footer/FooterText.js +0 -33
- package/lib/Subject/SubjectHeader.d.ts +0 -17
- package/lib/Subject/SubjectHeader.js +0 -73
- package/src/Footer/EditorName.tsx +0 -19
- package/src/Footer/Footer.tsx +0 -107
- package/src/Footer/FooterLinks.tsx +0 -144
- package/src/Footer/FooterText.tsx +0 -35
- package/src/Subject/SubjectHeader.tsx +0 -100
package/es/Embed/ImageEmbed.js
CHANGED
|
@@ -86,7 +86,7 @@ const expandedSizes = "(min-width: 1024px) 1024px, 100vw";
|
|
|
86
86
|
const StyledFigure = /*#__PURE__*/_styled(Figure, {
|
|
87
87
|
target: "ened8ka2",
|
|
88
88
|
label: "StyledFigure"
|
|
89
|
-
})("&:hover{[data-byline-button]{background:", colors.white, ";svg{transform:scale(1.2);}}}&[data-float=\"right\"]{float:right;}&[data-float=\"left\"]{float:left;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA8GmC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
89
|
+
})("&:hover{[data-byline-button]{background:", colors.white, ";svg{transform:scale(1.2);}}}&[data-float=\"right\"]{float:right;}&[data-float=\"left\"]{float:left;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA8GmC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
90
90
|
const ImageEmbed = _ref => {
|
|
91
91
|
let {
|
|
92
92
|
embed,
|
|
@@ -171,7 +171,7 @@ const ImageEmbed = _ref => {
|
|
|
171
171
|
const HiddenSpan = /*#__PURE__*/_styled("span", {
|
|
172
172
|
target: "ened8ka1",
|
|
173
173
|
label: "HiddenSpan"
|
|
174
|
-
})(utils.visuallyHidden, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AAqN8B","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
174
|
+
})(utils.visuallyHidden, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AAqN8B","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
175
175
|
const hideByline = size => {
|
|
176
176
|
return !!size && size.endsWith("-hide-byline");
|
|
177
177
|
};
|
|
@@ -202,7 +202,7 @@ const ImageWrapper = _ref2 => {
|
|
|
202
202
|
const BylineButton = /*#__PURE__*/_styled("button", {
|
|
203
203
|
target: "ened8ka0",
|
|
204
204
|
label: "BylineButton"
|
|
205
|
-
})("cursor:pointer;position:absolute;z-index:1;bottom:0;right:0;padding:", spacing.small, ";transition:all 0.3s ease-out;background:", colors.background.default, "20;border:0;svg{transition:transform 0.4s ease-out;width:", spacing.normal, ";height:", spacing.normal, ";fill:", colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA+PkC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
205
|
+
})("cursor:pointer;position:absolute;z-index:1;bottom:0;right:0;padding:", spacing.small, ";transition:all 0.3s ease-out;background:", colors.background.default, "20;border:0;svg{transition:transform 0.4s ease-out;width:", spacing.normal, ";height:", spacing.normal, ";fill:", colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA+PkC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
206
206
|
const ExpandButton = _ref3 => {
|
|
207
207
|
let {
|
|
208
208
|
size,
|
|
@@ -218,6 +218,7 @@ const ExpandButton = _ref3 => {
|
|
|
218
218
|
return _jsx("button", {
|
|
219
219
|
type: "button",
|
|
220
220
|
css: figureActionIndicatorStyle,
|
|
221
|
+
"data-byline-button": "",
|
|
221
222
|
"aria-label": t("license.images.itemImage.zoom".concat(expanded ? "Out" : "", "ImageButtonLabel")),
|
|
222
223
|
onClick: onExpand,
|
|
223
224
|
children: expanded ? _jsx(ArrowCollapse, {}) : _jsx(ExpandTwoArrows, {})
|
package/es/Embed/index.js
CHANGED
|
@@ -6,7 +6,7 @@
|
|
|
6
6
|
*
|
|
7
7
|
*/
|
|
8
8
|
|
|
9
|
-
export { default as ImageEmbed } from "./ImageEmbed";
|
|
9
|
+
export { default as ImageEmbed, getCrop, getFocalPoint } from "./ImageEmbed";
|
|
10
10
|
export { default as AudioEmbed } from "./AudioEmbed";
|
|
11
11
|
export { default as H5pEmbed } from "./H5pEmbed";
|
|
12
12
|
export { default as ExternalEmbed } from "./ExternalEmbed";
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import _styled from "@emotion/styled/base";
|
|
2
|
+
/**
|
|
3
|
+
* Copyright (c) 2016-present, NDLA.
|
|
4
|
+
*
|
|
5
|
+
* This source code is licensed under the GPLv3 license found in the
|
|
6
|
+
* LICENSE file in the root directory of this source tree.
|
|
7
|
+
*
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { breakpoints, colors, mq, spacing, stackOrder } from "@ndla/core";
|
|
11
|
+
export const FooterBlock = /*#__PURE__*/_styled("div", {
|
|
12
|
+
target: "eu3vydv0",
|
|
13
|
+
label: "FooterBlock"
|
|
14
|
+
})("position:relative;display:flex;flex-direction:column;align-items:center;gap:", spacing.large, ";color:", colors.white, ";z-index:", stackOrder.offsetSingle, ";padding:", spacing.large, " ", spacing.large, " ", spacing.xlarge, ";background:", colors.brand.dark, ";", mq.range({
|
|
15
|
+
from: breakpoints.wide
|
|
16
|
+
}), "{padding:", spacing.large, " 170px ", spacing.xlarge, ";}>*{max-width:1100px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkZvb3RlckJsb2NrLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFXcUMiLCJmaWxlIjoiRm9vdGVyQmxvY2sudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTYtcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgc3R5bGVkIGZyb20gXCJAZW1vdGlvbi9zdHlsZWRcIjtcbmltcG9ydCB7IGJyZWFrcG9pbnRzLCBjb2xvcnMsIG1xLCBzcGFjaW5nLCBzdGFja09yZGVyIH0gZnJvbSBcIkBuZGxhL2NvcmVcIjtcblxuZXhwb3J0IGNvbnN0IEZvb3RlckJsb2NrID0gc3R5bGVkLmRpdmBcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xuICBkaXNwbGF5OiBmbGV4O1xuICBmbGV4LWRpcmVjdGlvbjogY29sdW1uO1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBnYXA6ICR7c3BhY2luZy5sYXJnZX07XG4gIGNvbG9yOiAke2NvbG9ycy53aGl0ZX07XG4gIHotaW5kZXg6ICR7c3RhY2tPcmRlci5vZmZzZXRTaW5nbGV9O1xuICBwYWRkaW5nOiAke3NwYWNpbmcubGFyZ2V9ICR7c3BhY2luZy5sYXJnZX0gJHtzcGFjaW5nLnhsYXJnZX07XG4gIGJhY2tncm91bmQ6ICR7Y29sb3JzLmJyYW5kLmRhcmt9O1xuICAke21xLnJhbmdlKHsgZnJvbTogYnJlYWtwb2ludHMud2lkZSB9KX0ge1xuICAgIHBhZGRpbmc6ICR7c3BhY2luZy5sYXJnZX0gMTcwcHggJHtzcGFjaW5nLnhsYXJnZX07XG4gIH1cbiAgPiAqIHtcbiAgICBtYXgtd2lkdGg6IDExMDBweDtcbiAgfVxuYDtcblxuZXhwb3J0IGRlZmF1bHQgRm9vdGVyQmxvY2s7XG4iXX0= */"));
|
|
17
|
+
export default FooterBlock;
|
package/es/Footer/index.js
CHANGED
package/es/Subject/index.js
CHANGED
package/es/index.js
CHANGED
|
@@ -11,7 +11,7 @@
|
|
|
11
11
|
|
|
12
12
|
export { ExpandableBox, ExpandableBoxSummary } from "./ExpandableBox";
|
|
13
13
|
export { default as FramedContent } from "./FramedContent";
|
|
14
|
-
export { ConceptNotionV2, ImageEmbed, AudioEmbed, H5pEmbed, ExternalEmbed, IframeEmbed, FootnoteEmbed, BrightcoveEmbed, ContentLinkEmbed, RelatedContentEmbed, ConceptEmbed, ConceptListEmbed, UnknownEmbed, InlineConcept, BlockConcept, UuDisclaimerEmbed, CopyrightEmbed, CodeEmbed } from "./Embed";
|
|
14
|
+
export { ConceptNotionV2, ImageEmbed, getCrop, getFocalPoint, AudioEmbed, H5pEmbed, ExternalEmbed, IframeEmbed, FootnoteEmbed, BrightcoveEmbed, ContentLinkEmbed, RelatedContentEmbed, ConceptEmbed, ConceptListEmbed, UnknownEmbed, InlineConcept, BlockConcept, UuDisclaimerEmbed, CopyrightEmbed, CodeEmbed } from "./Embed";
|
|
15
15
|
export { LicenseLink } from "./LicenseByline";
|
|
16
16
|
export { ArticleByline, ArticleFootNotes, ArticleIntroduction, ArticleTitle, ArticleWrapper, ArticleHeaderWrapper, default as Article } from "./Article";
|
|
17
17
|
export { getPossiblyRelativeUrl } from "./utils/relativeUrl";
|
|
@@ -28,7 +28,7 @@ export { default as FactBox } from "./FactBox";
|
|
|
28
28
|
export { default as Image, ImageLink, makeSrcQueryString } from "./Image";
|
|
29
29
|
export { SubjectMaterialHero, TasksAndActivitiesHero, AssessmentResourcesHero, SubjectHero, SourceMaterialHero, Hero, NdlaFilmHero, ConceptHero, HeroContent } from "./Hero";
|
|
30
30
|
export { FilterButtons, ToggleItem } from "./Filter";
|
|
31
|
-
export {
|
|
31
|
+
export { FooterBlock } from "./Footer";
|
|
32
32
|
export { Figure } from "./Figure";
|
|
33
33
|
export { LanguageSelector } from "./LanguageSelector";
|
|
34
34
|
export { LearningPathWrapper, LearningPathContent, LearningPathMenu, LearningPathSticky, LearningPathInformation, LearningPathStickySibling, LearningPathStickyPlaceholder, LearningPathLastStepNavigation, LearningPathMobileStepInfo, LearningPathMobileHeader } from "./LearningPaths";
|
|
@@ -60,7 +60,7 @@ export { default as LayoutItem, OneColumn, PageContainer, Content } from "./Layo
|
|
|
60
60
|
export { FilmSlideshow, MovieGrid, FilmMovieSearch, FilmMovieList, AllMoviesAlphabetically } from "./NDLAFilm";
|
|
61
61
|
export { MediaList, MediaListItem, MediaListItemBody, MediaListItemActions, MediaListItemImage, MediaListItemMeta } from "./MediaList";
|
|
62
62
|
export { default as ContentTypeBadge, SubjectMaterialBadge, TasksAndActivitiesBadge, AssessmentResourcesBadge, LearningPathBadge, SubjectBadge, SourceMaterialBadge, ConceptBadge } from "./ContentTypeBadge";
|
|
63
|
-
export {
|
|
63
|
+
export { SubjectBanner } from "./Subject";
|
|
64
64
|
export { default as CopyParagraphButton } from "./CopyParagraphButton";
|
|
65
65
|
export { default as ContentPlaceholder } from "./ContentPlaceholder";
|
|
66
66
|
export { Notion } from "./Notion";
|
package/es/locale/messages-en.js
CHANGED
package/es/locale/messages-nb.js
CHANGED
package/es/locale/messages-nn.js
CHANGED
package/es/locale/messages-se.js
CHANGED
package/lib/Embed/ImageEmbed.js
CHANGED
|
@@ -98,7 +98,7 @@ const expandedSizes = "(min-width: 1024px) 1024px, 100vw";
|
|
|
98
98
|
const StyledFigure = /*#__PURE__*/(0, _base.default)(_Figure.Figure, {
|
|
99
99
|
target: "ened8ka2",
|
|
100
100
|
label: "StyledFigure"
|
|
101
|
-
})("&:hover{[data-byline-button]{background:", _core.colors.white, ";svg{transform:scale(1.2);}}}&[data-float=\"right\"]{float:right;}&[data-float=\"left\"]{float:left;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA8GmC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
101
|
+
})("&:hover{[data-byline-button]{background:", _core.colors.white, ";svg{transform:scale(1.2);}}}&[data-float=\"right\"]{float:right;}&[data-float=\"left\"]{float:left;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA8GmC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
102
102
|
const ImageEmbed = _ref => {
|
|
103
103
|
let {
|
|
104
104
|
embed,
|
|
@@ -183,7 +183,7 @@ const ImageEmbed = _ref => {
|
|
|
183
183
|
const HiddenSpan = /*#__PURE__*/(0, _base.default)("span", {
|
|
184
184
|
target: "ened8ka1",
|
|
185
185
|
label: "HiddenSpan"
|
|
186
|
-
})(_core.utils.visuallyHidden, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AAqN8B","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
186
|
+
})(_core.utils.visuallyHidden, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AAqN8B","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
187
187
|
const hideByline = size => {
|
|
188
188
|
return !!size && size.endsWith("-hide-byline");
|
|
189
189
|
};
|
|
@@ -214,7 +214,7 @@ const ImageWrapper = _ref2 => {
|
|
|
214
214
|
const BylineButton = /*#__PURE__*/(0, _base.default)("button", {
|
|
215
215
|
target: "ened8ka0",
|
|
216
216
|
label: "BylineButton"
|
|
217
|
-
})("cursor:pointer;position:absolute;z-index:1;bottom:0;right:0;padding:", _core.spacing.small, ";transition:all 0.3s ease-out;background:", _core.colors.background.default, "20;border:0;svg{transition:transform 0.4s ease-out;width:", _core.spacing.normal, ";height:", _core.spacing.normal, ";fill:", _core.colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA+PkC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
217
|
+
})("cursor:pointer;position:absolute;z-index:1;bottom:0;right:0;padding:", _core.spacing.small, ";transition:all 0.3s ease-out;background:", _core.colors.background.default, "20;border:0;svg{transition:transform 0.4s ease-out;width:", _core.spacing.normal, ";height:", _core.spacing.normal, ";fill:", _core.colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA+PkC","file":"ImageEmbed.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 { MouseEventHandler, ReactNode, useMemo, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { colors, spacing, utils } from \"@ndla/core\";\nimport { ExpandTwoArrows } from \"@ndla/icons/action\";\nimport { ArrowCollapse, ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from \"./types\";\nimport { Figure, FigureType, figureActionIndicatorStyle } from \"../Figure\";\nimport Image, { ImageLink } from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs[\"image\"];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n  children?: ReactNode;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is \"xsmall\" | \"small\" => size === \"xsmall\" || size === \"small\";\n\nconst isAlign = (align?: string): align is \"left\" | \"right\" => align === \"left\" || align === \"right\";\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return \"full\";\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === \"full\") {\n    return \"(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw\";\n  }\n  if (align && size === \"small\") {\n    return \"(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw\";\n  }\n  if (align && size === \"xsmall\") {\n    return \"(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw\";\n  }\n  return \"(min-width: 1024px) 1024px, 100vw\";\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? \"\");\n  const focalY = parseFloat(data.focalY ?? \"\");\n  if (!isNaN(focalX) && !isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = parseFloat(data.upperLeftY ?? \"\");\n  if (!isNaN(lowerRightX) && !isNaN(lowerRightY) && !isNaN(upperLeftX) && !isNaN(upperLeftY)) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = \"(min-width: 1024px) 1024px, 100vw\";\n\nconst StyledFigure = styled(Figure)`\n  &:hover {\n    [data-byline-button] {\n      background: ${colors.white};\n      svg {\n        transform: scale(1.2);\n      }\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = \"article\",\n  children,\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === \"success\" && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === \"error\") {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={\"image\"} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || \"\";\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} data-float={embedData.align}>\n      {children}\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          hideOnLargeScreens={isSmall(embedData.size) && !imageSizes}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </StyledFigure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith(\"-hide-byline\");\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t(\"license.images.itemImage.ariaLabel\")}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst BylineButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  z-index: 1;\n  bottom: 0;\n  right: 0;\n  padding: ${spacing.small};\n  transition: all 0.3s ease-out;\n  background: ${colors.background.default}20;\n  border: 0;\n\n  svg {\n    transition: transform 0.4s ease-out;\n    width: ${spacing.normal};\n    height: ${spacing.normal};\n    fill: ${colors.brand.primary};\n  }\n`;\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        css={figureActionIndicatorStyle}\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <BylineButton\n        type=\"button\"\n        data-byline-button=\"\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? \"expandByline\" : \"minimizeByline\"}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </BylineButton>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
|
|
218
218
|
const ExpandButton = _ref3 => {
|
|
219
219
|
let {
|
|
220
220
|
size,
|
|
@@ -230,6 +230,7 @@ const ExpandButton = _ref3 => {
|
|
|
230
230
|
return (0, _jsxRuntime.jsx)("button", {
|
|
231
231
|
type: "button",
|
|
232
232
|
css: _Figure.figureActionIndicatorStyle,
|
|
233
|
+
"data-byline-button": "",
|
|
233
234
|
"aria-label": t("license.images.itemImage.zoom".concat(expanded ? "Out" : "", "ImageButtonLabel")),
|
|
234
235
|
onClick: onExpand,
|
|
235
236
|
children: expanded ? (0, _jsxRuntime.jsx)(_common.ArrowCollapse, {}) : (0, _jsxRuntime.jsx)(_action.ExpandTwoArrows, {})
|
package/lib/Embed/index.d.ts
CHANGED
|
@@ -5,7 +5,7 @@
|
|
|
5
5
|
* LICENSE file in the root directory of this source tree.
|
|
6
6
|
*
|
|
7
7
|
*/
|
|
8
|
-
export { default as ImageEmbed } from "./ImageEmbed";
|
|
8
|
+
export { default as ImageEmbed, getCrop, getFocalPoint } from "./ImageEmbed";
|
|
9
9
|
export { default as AudioEmbed } from "./AudioEmbed";
|
|
10
10
|
export { default as H5pEmbed } from "./H5pEmbed";
|
|
11
11
|
export { default as ExternalEmbed } from "./ExternalEmbed";
|
package/lib/Embed/index.js
CHANGED
|
@@ -111,7 +111,19 @@ Object.defineProperty(exports, "UuDisclaimerEmbed", {
|
|
|
111
111
|
return _UuDisclaimerEmbed.default;
|
|
112
112
|
}
|
|
113
113
|
});
|
|
114
|
-
|
|
114
|
+
Object.defineProperty(exports, "getCrop", {
|
|
115
|
+
enumerable: true,
|
|
116
|
+
get: function () {
|
|
117
|
+
return _ImageEmbed.getCrop;
|
|
118
|
+
}
|
|
119
|
+
});
|
|
120
|
+
Object.defineProperty(exports, "getFocalPoint", {
|
|
121
|
+
enumerable: true,
|
|
122
|
+
get: function () {
|
|
123
|
+
return _ImageEmbed.getFocalPoint;
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
var _ImageEmbed = _interopRequireWildcard(require("./ImageEmbed"));
|
|
115
127
|
var _AudioEmbed = _interopRequireDefault(require("./AudioEmbed"));
|
|
116
128
|
var _H5pEmbed = _interopRequireDefault(require("./H5pEmbed"));
|
|
117
129
|
var _ExternalEmbed = _interopRequireDefault(require("./ExternalEmbed"));
|
|
@@ -127,6 +139,6 @@ var _UnknownEmbed = _interopRequireDefault(require("./UnknownEmbed"));
|
|
|
127
139
|
var _UuDisclaimerEmbed = _interopRequireDefault(require("./UuDisclaimerEmbed"));
|
|
128
140
|
var _CopyrightEmbed = _interopRequireDefault(require("./CopyrightEmbed"));
|
|
129
141
|
var _CodeEmbed = _interopRequireDefault(require("./CodeEmbed"));
|
|
142
|
+
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
130
143
|
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
|
|
131
|
-
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
|
|
132
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
144
|
+
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
|