@ndla/ui 55.0.5-alpha.0 → 55.0.6-alpha.1

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.
@@ -87,7 +87,7 @@ const expandedSizes = "(min-width: 1024px) 1024px, 100vw";
87
87
  const StyledFigure = /*#__PURE__*/_styled(Figure, {
88
88
  target: "ened8ka2",
89
89
  label: "StyledFigure"
90
- })("&:hover{[data-byline-button]{background:", colors.white, ";}button[data-expanded]{transform:scale(1.2);}}button[data-expanded=\"true\"]{svg{transform:rotate(-45deg);}}&[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":"AAiHmC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  line-height: 1;\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
90
+ })("&:hover{[data-byline-button]{background:", colors.white, ";}button[data-expanded]{transform:scale(1.2);}}button[data-expanded=\"true\"]{svg{transform:rotate(-45deg);}}&[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":"AAiHmC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
91
91
  const ImageEmbed = _ref => {
92
92
  let {
93
93
  embed,
@@ -182,11 +182,11 @@ const hideByline = embed => {
182
182
  const BylineButton = /*#__PURE__*/_styled("button", {
183
183
  target: "ened8ka1",
184
184
  label: "BylineButton"
185
- })("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":"AAyOkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  line-height: 1;\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
185
+ })("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":"AAyOkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
186
186
  const StyledButton = /*#__PURE__*/_styled("button", {
187
187
  target: "ened8ka0",
188
188
  label: "StyledButton"
189
- })("cursor:pointer;position:absolute;padding:0;top:", spacing.small, ";right:", spacing.small, ";width:", spacing.mediumlarge, ";height:", spacing.mediumlarge, ";border:2px solid ", colors.white, ";transition:all 0.3s ease-out;color:", colors.white, ";background-color:", colors.brand.primary, ";border-radius:", misc.borderRadiusLarge, ";line-height:1;svg{transition:transform 0.4s ease-out;height:", spacing.medium, ";width:", spacing.medium, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA4PkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  line-height: 1;\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
189
+ })("display:flex;align-items:center;justify-content:center;cursor:pointer;position:absolute;padding:0;top:", spacing.small, ";right:", spacing.small, ";width:", spacing.mediumlarge, ";height:", spacing.mediumlarge, ";border:2px solid ", colors.white, ";transition:all 0.3s ease-out;color:", colors.white, ";background-color:", colors.brand.primary, ";border-radius:", misc.borderRadiusLarge, ";svg{transition:transform 0.4s ease-out;height:", spacing.medium, ";width:", spacing.medium, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA4PkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
190
190
  const ExpandButton = _ref2 => {
191
191
  let {
192
192
  align,
@@ -23,13 +23,15 @@ export const getPossiblyRelativeUrl = (url, path) => {
23
23
  // If the host is the same, return the relative path
24
24
  if (urlObj.hostname.replace(REPLACE_WWW, "") === pathObj.hostname.replace(REPLACE_WWW, "")) {
25
25
  // Replace the language part of the url with the language part of the path
26
+ // Keep the search params if they exist
27
+ const search = urlObj.search;
26
28
  // If the path language part does not exist, remove it.
27
29
  const urlMatch = urlObj.pathname.match(LANGUAGE_REGEX);
28
30
  const pathMatch = pathObj.pathname.match(LANGUAGE_REGEX);
29
31
  if (urlMatch !== null && urlMatch !== void 0 && urlMatch[1] && (urlMatch === null || urlMatch === void 0 ? void 0 : urlMatch[1]) !== (pathMatch === null || pathMatch === void 0 ? void 0 : pathMatch[1])) {
30
- return urlObj.pathname.replace(urlMatch[1], (pathMatch === null || pathMatch === void 0 ? void 0 : pathMatch[1]) || "");
32
+ return "".concat(urlObj.pathname.replace(urlMatch[1], (pathMatch === null || pathMatch === void 0 ? void 0 : pathMatch[1]) || "")).concat(search);
31
33
  }
32
- return urlObj.pathname;
34
+ return "".concat(urlObj.pathname).concat(search);
33
35
  }
34
36
  return url;
35
37
  };
@@ -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, ";}button[data-expanded]{transform:scale(1.2);}}button[data-expanded=\"true\"]{svg{transform:rotate(-45deg);}}&[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":"AAiHmC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  line-height: 1;\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
101
+ })("&:hover{[data-byline-button]{background:", _core.colors.white, ";}button[data-expanded]{transform:scale(1.2);}}button[data-expanded=\"true\"]{svg{transform:rotate(-45deg);}}&[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":"AAiHmC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
102
102
  const ImageEmbed = _ref => {
103
103
  let {
104
104
  embed,
@@ -193,11 +193,11 @@ const hideByline = embed => {
193
193
  const BylineButton = /*#__PURE__*/(0, _base.default)("button", {
194
194
  target: "ened8ka1",
195
195
  label: "BylineButton"
196
- })("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":"AAyOkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  line-height: 1;\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
196
+ })("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":"AAyOkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
197
197
  const StyledButton = /*#__PURE__*/(0, _base.default)("button", {
198
198
  target: "ened8ka0",
199
199
  label: "StyledButton"
200
- })("cursor:pointer;position:absolute;padding:0;top:", _core.spacing.small, ";right:", _core.spacing.small, ";width:", _core.spacing.mediumlarge, ";height:", _core.spacing.mediumlarge, ";border:2px solid ", _core.colors.white, ";transition:all 0.3s ease-out;color:", _core.colors.white, ";background-color:", _core.colors.brand.primary, ";border-radius:", _core.misc.borderRadiusLarge, ";line-height:1;svg{transition:transform 0.4s ease-out;height:", _core.spacing.medium, ";width:", _core.spacing.medium, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA4PkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  line-height: 1;\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
200
+ })("display:flex;align-items:center;justify-content:center;cursor:pointer;position:absolute;padding:0;top:", _core.spacing.small, ";right:", _core.spacing.small, ";width:", _core.spacing.mediumlarge, ";height:", _core.spacing.mediumlarge, ";border:2px solid ", _core.colors.white, ";transition:all 0.3s ease-out;color:", _core.colors.white, ";background-color:", _core.colors.brand.primary, ";border-radius:", _core.misc.borderRadiusLarge, ";svg{transition:transform 0.4s ease-out;height:", _core.spacing.medium, ";width:", _core.spacing.medium, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AA4PkC","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\n/** @jsxImportSource @emotion/react */\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, misc, spacing } from \"@ndla/core\";\nimport { Plus } from \"@ndla/icons/action\";\nimport { ChevronDown, ChevronUp } from \"@ndla/icons/common\";\nimport { COPYRIGHTED } from \"@ndla/licenses\";\nimport { ImageEmbedData, ImageMetaData } from \"@ndla/types-embed\";\nimport EmbedErrorPlaceholder from \"./EmbedErrorPlaceholder\";\nimport { RenderContext } from \"./types\";\nimport { Figure, FigureType } from \"../Figure\";\nimport Image from \"../Image\";\nimport { EmbedByline } from \"../LicenseByline\";\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\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 = Number.parseFloat(data.focalX ?? \"\");\n  const focalY = Number.parseFloat(data.focalY ?? \"\");\n  if (!Number.isNaN(focalX) && !Number.isNaN(focalY)) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = Number.parseFloat(data.lowerRightX ?? \"\");\n  const lowerRightY = Number.parseFloat(data.lowerRightY ?? \"\");\n  const upperLeftX = Number.parseFloat(data.upperLeftX ?? \"\");\n  const upperLeftY = Number.parseFloat(data.upperLeftY ?? \"\");\n  if (\n    !Number.isNaN(lowerRightX) &&\n    !Number.isNaN(lowerRightY) &&\n    !Number.isNaN(upperLeftX) &&\n    !Number.isNaN(upperLeftY)\n  ) {\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    }\n    button[data-expanded] {\n      transform: scale(1.2);\n    }\n  }\n  button[data-expanded=\"true\"] {\n    svg {\n      transform: rotate(-45deg);\n    }\n  }\n  &[data-float=\"right\"] {\n    float: right;\n  }\n  &[data-float=\"left\"] {\n    float: left;\n  }\n`;\n\nconst ImageEmbed = ({ embed, previewAlt, inGrid, lang, renderContext = \"article\", children }: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n  // Full-size figures automatically get a margin of {spacing.normal} on its y-axis if a float is not set (or if float is an empty string).\n  // This adds some margin to normal figures within an article, but should not happen for figures in a grid.\n  const [floatAttr, setFloatAttr] = useState<{ \"data-float\"?: string }>(() =>\n    inGrid && !embed.embedData.align ? {} : { \"data-float\": embed.embedData.align },\n  );\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === \"article\") {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    }\n    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  const toggleImageSize = () => {\n    if (!imageSizes) {\n      setImageSizes(expandedSizes);\n      setTimeout(() => {\n        setFloatAttr({});\n      }, 400); //Removing the float parameter too quickly causes the image to be resized from left regardless\n    } else {\n      setImageSizes(undefined);\n      setFloatAttr({ \"data-float\": embedData.align });\n    }\n  };\n\n  return (\n    <StyledFigure type={imageSizes ? undefined : figureType} {...floatAttr}>\n      {children}\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        onExpand={isAlign(embedData.align) ? toggleImageSize : undefined}\n        expanded={!!imageSizes}\n        expandButton={\n          <ExpandButton\n            embedData={embedData}\n            expanded={!!imageSizes}\n            align={embedData.align}\n            bylineHidden={isBylineHidden}\n            onExpand={toggleImageSize}\n            onHideByline={() => setIsBylineHidden((p) => !p)}\n          />\n        }\n        lang={lang}\n      />\n      {isBylineHidden ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          visibleAlt={previewAlt ? embed.embedData.alt : \"\"}\n        />\n      )}\n    </StyledFigure>\n  );\n};\n\nconst hideByline = (embed: ImageEmbedData): boolean => {\n  return (!!embed.size && embed.size.endsWith(\"-hide-byline\")) || embed.hideByline === \"true\";\n};\n\ninterface ExpandButtonProps {\n  embedData: ImageEmbedData;\n  align?: 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 StyledButton = styled.button`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  cursor: pointer;\n  position: absolute;\n  padding: 0;\n  top: ${spacing.small};\n  right: ${spacing.small};\n  width: ${spacing.mediumlarge};\n  height: ${spacing.mediumlarge};\n  border: 2px solid ${colors.white};\n  transition: all 0.3s ease-out;\n  color: ${colors.white};\n  background-color: ${colors.brand.primary};\n  border-radius: ${misc.borderRadiusLarge};\n  svg {\n    transition: transform 0.4s ease-out;\n    height: ${spacing.medium};\n    width: ${spacing.medium};\n  }\n`;\n\nconst ExpandButton = ({ align, embedData, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isAlign(align)) {\n    return (\n      <StyledButton\n        type=\"button\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? \"Out\" : \"\"}ImageButtonLabel`)}\n        onClick={onExpand}\n        data-expanded={expanded}\n      >\n        <Plus />\n      </StyledButton>\n    );\n  }\n  if (hideByline(embedData)) {\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  }\n  return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
201
201
  const ExpandButton = _ref2 => {
202
202
  let {
203
203
  align,
@@ -29,13 +29,15 @@ const getPossiblyRelativeUrl = (url, path) => {
29
29
  // If the host is the same, return the relative path
30
30
  if (urlObj.hostname.replace(REPLACE_WWW, "") === pathObj.hostname.replace(REPLACE_WWW, "")) {
31
31
  // Replace the language part of the url with the language part of the path
32
+ // Keep the search params if they exist
33
+ const search = urlObj.search;
32
34
  // If the path language part does not exist, remove it.
33
35
  const urlMatch = urlObj.pathname.match(LANGUAGE_REGEX);
34
36
  const pathMatch = pathObj.pathname.match(LANGUAGE_REGEX);
35
37
  if (urlMatch !== null && urlMatch !== void 0 && urlMatch[1] && (urlMatch === null || urlMatch === void 0 ? void 0 : urlMatch[1]) !== (pathMatch === null || pathMatch === void 0 ? void 0 : pathMatch[1])) {
36
- return urlObj.pathname.replace(urlMatch[1], (pathMatch === null || pathMatch === void 0 ? void 0 : pathMatch[1]) || "");
38
+ return "".concat(urlObj.pathname.replace(urlMatch[1], (pathMatch === null || pathMatch === void 0 ? void 0 : pathMatch[1]) || "")).concat(search);
37
39
  }
38
- return urlObj.pathname;
40
+ return "".concat(urlObj.pathname).concat(search);
39
41
  }
40
42
  return url;
41
43
  };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ndla/ui",
3
- "version": "55.0.5-alpha.0",
3
+ "version": "55.0.6-alpha.1",
4
4
  "description": "UI component library for NDLA.",
5
5
  "license": "GPL-3.0",
6
6
  "main": "lib/index.js",
@@ -76,5 +76,5 @@
76
76
  "publishConfig": {
77
77
  "access": "public"
78
78
  },
79
- "gitHead": "70958e217cbb56d76045d175c150bc755bda4aee"
79
+ "gitHead": "63a1e8dbc12f6d268aa3140f408d1b1241c36a80"
80
80
  }
@@ -251,6 +251,9 @@ const BylineButton = styled.button`
251
251
  `;
252
252
 
253
253
  const StyledButton = styled.button`
254
+ display: flex;
255
+ align-items: center;
256
+ justify-content: center;
254
257
  cursor: pointer;
255
258
  position: absolute;
256
259
  padding: 0;
@@ -263,7 +266,6 @@ const StyledButton = styled.button`
263
266
  color: ${colors.white};
264
267
  background-color: ${colors.brand.primary};
265
268
  border-radius: ${misc.borderRadiusLarge};
266
- line-height: 1;
267
269
  svg {
268
270
  transition: transform 0.4s ease-out;
269
271
  height: ${spacing.medium};
@@ -75,4 +75,16 @@ describe("getPossibleRelativeUrl", () => {
75
75
 
76
76
  expect(getPossiblyRelativeUrl(url, pathname)).toEqual("mailto:test@ndla.no");
77
77
  });
78
+ it("handles params in url", () => {
79
+ const url = "https://ndla.no/search?grepCodes=KM123";
80
+ const pathname = "https://ndla.no/article/666";
81
+
82
+ expect(getPossiblyRelativeUrl(url, pathname)).toEqual("/search?grepCodes=KM123");
83
+ });
84
+ it("handles params in url including language tag", () => {
85
+ const url = "https://ndla.no/nb/search?grepCodes=KM123";
86
+ const pathname = "https://ndla.no/en/article/666";
87
+
88
+ expect(getPossiblyRelativeUrl(url, pathname)).toEqual("/en/search?grepCodes=KM123");
89
+ });
78
90
  });
@@ -26,14 +26,16 @@ export const getPossiblyRelativeUrl = (url: string, path?: string) => {
26
26
  // If the host is the same, return the relative path
27
27
  if (urlObj.hostname.replace(REPLACE_WWW, "") === pathObj.hostname.replace(REPLACE_WWW, "")) {
28
28
  // Replace the language part of the url with the language part of the path
29
+ // Keep the search params if they exist
30
+ const search = urlObj.search;
29
31
  // If the path language part does not exist, remove it.
30
32
  const urlMatch = urlObj.pathname.match(LANGUAGE_REGEX);
31
33
  const pathMatch = pathObj.pathname.match(LANGUAGE_REGEX);
32
34
  if (urlMatch?.[1] && urlMatch?.[1] !== pathMatch?.[1]) {
33
- return urlObj.pathname.replace(urlMatch[1], pathMatch?.[1] || "");
35
+ return `${urlObj.pathname.replace(urlMatch[1], pathMatch?.[1] || "")}${search}`;
34
36
  }
35
37
 
36
- return urlObj.pathname;
38
+ return `${urlObj.pathname}${search}`;
37
39
  }
38
40
  return url;
39
41
  };