@ndla/ui 49.0.6 → 49.0.7

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.
@@ -6,12 +6,13 @@
6
6
  *
7
7
  */
8
8
  import { ImageEmbedData, ImageMetaData } from '@ndla/types-embed';
9
- import { HeartButtonType, RenderContext } from './types';
9
+ import { CanonicalUrlFuncs, HeartButtonType, RenderContext } from './types';
10
10
  interface Props {
11
11
  embed: ImageMetaData;
12
12
  previewAlt?: boolean;
13
13
  path?: string;
14
14
  heartButton?: HeartButtonType;
15
+ canonicalUrl?: CanonicalUrlFuncs['image'];
15
16
  inGrid?: boolean;
16
17
  lang?: string;
17
18
  renderContext?: RenderContext;
@@ -40,5 +41,5 @@ export declare const getCrop: (data: ImageEmbedData) => {
40
41
  endX: number;
41
42
  endY: number;
42
43
  } | undefined;
43
- declare const ImageEmbed: ({ embed, previewAlt, heartButton: HeartButton, inGrid, path, lang, renderContext, }: Props) => import("@emotion/react/jsx-runtime").JSX.Element;
44
+ declare const ImageEmbed: ({ embed, previewAlt, heartButton: HeartButton, inGrid, path, lang, canonicalUrl, renderContext, }: Props) => import("@emotion/react/jsx-runtime").JSX.Element;
44
45
  export default ImageEmbed;
@@ -112,6 +112,7 @@ var ImageEmbed = function ImageEmbed(_ref) {
112
112
  inGrid = _ref.inGrid,
113
113
  path = _ref.path,
114
114
  lang = _ref.lang,
115
+ canonicalUrl = _ref.canonicalUrl,
115
116
  _ref$renderContext = _ref.renderContext,
116
117
  renderContext = _ref$renderContext === void 0 ? 'article' : _ref$renderContext;
117
118
  var _useState = (0, _react.useState)(hideByline(embed.embedData.size)),
@@ -151,7 +152,7 @@ var ImageEmbed = function ImageEmbed(_ref) {
151
152
  type: imageSizes ? undefined : figureType,
152
153
  className: imageSizes ? "c-figure--".concat(embedData.align, " expanded") : '',
153
154
  children: [(0, _jsxRuntime.jsx)(ImageWrapper, {
154
- src: !isCopyrighted ? embedData.pageUrl || data.image.imageUrl : undefined,
155
+ src: !isCopyrighted ? canonicalUrl === null || canonicalUrl === void 0 ? void 0 : canonicalUrl(data) : undefined,
155
156
  crop: crop,
156
157
  size: embedData.size,
157
158
  pagePath: path,
@@ -196,7 +197,7 @@ var ImageEmbed = function ImageEmbed(_ref) {
196
197
  var HiddenSpan = /*#__PURE__*/(0, _base.default)("span", {
197
198
  target: "ened8ka0",
198
199
  label: "HiddenSpan"
199
- })(_core.utils.visuallyHidden, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AAiM8B","file":"ImageEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ImageEmbedData, ImageMetaData } from '@ndla/types-embed';\nimport { useTranslation } from 'react-i18next';\nimport { MouseEventHandler, useMemo, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { ExpandTwoArrows } from '@ndla/icons/action';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { ArrowCollapse, ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { utils } from '@ndla/core';\nimport styled from '@emotion/styled';\nimport { Figure, FigureType } from '../Figure';\nimport Image, { ImageLink } from '../Image';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { HeartButtonType, RenderContext } from './types';\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is 'xsmall' | 'small' => size === 'xsmall' || size === 'small';\n\nconst isAlign = (align?: string): align is 'left' | 'right' => align === 'left' || align === 'right';\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return 'full';\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === 'full') {\n    return '(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw';\n  }\n  if (align && size === 'small') {\n    return '(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw';\n  }\n  if (align && size === 'xsmall') {\n    return '(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw';\n  }\n  return '(min-width: 1024px) 1024px, 100vw';\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? '');\n  const focalY = parseFloat(data.focalY ?? '');\n  if (!!focalX && !!focalY) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? '');\n  const lowerRightY = parseFloat(data.lowerRightY ?? '');\n  const upperLeftX = parseFloat(data.upperLeftX ?? '');\n  const upperLeftY = parseFloat(data.upperLeftY ?? '');\n  if (!!lowerRightX && !!lowerRightY && !!upperLeftX && !!upperLeftY) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = '(min-width: 1024px) 1024px, 100vw';\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  renderContext = 'article',\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === 'article') {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === 'success' && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === 'error') {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={'image'} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || '';\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <Figure\n      type={imageSizes ? undefined : figureType}\n      className={imageSizes ? `c-figure--${embedData.align} expanded` : ''}\n    >\n      <ImageWrapper\n        src={!isCopyrighted ? embedData.pageUrl || data.image.imageUrl : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden || (isSmall(embedData.size) && !imageSizes) ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : ''}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </Figure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith('-hide-byline');\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t('license.images.itemImage.ariaLabel')}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        className=\"c-figure__fullscreen-btn\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? 'Out' : ''}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <button\n        type=\"button\"\n        className=\"c-figure__show-byline-btn\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? 'expandByline' : 'minimizeByline'}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </button>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
200
+ })(_core.utils.visuallyHidden, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ImageEmbed.tsx"],"names":[],"mappings":"AAmM8B","file":"ImageEmbed.tsx","sourcesContent":["/**\n * Copyright (c) 2023-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { ImageEmbedData, ImageMetaData } from '@ndla/types-embed';\nimport { useTranslation } from 'react-i18next';\nimport { MouseEventHandler, useMemo, useState } from 'react';\nimport parse from 'html-react-parser';\nimport { ExpandTwoArrows } from '@ndla/icons/action';\nimport { COPYRIGHTED } from '@ndla/licenses';\nimport { ArrowCollapse, ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { utils } from '@ndla/core';\nimport styled from '@emotion/styled';\nimport { Figure, FigureType } from '../Figure';\nimport Image, { ImageLink } from '../Image';\nimport { EmbedByline } from '../LicenseByline';\nimport EmbedErrorPlaceholder from './EmbedErrorPlaceholder';\nimport { CanonicalUrlFuncs, HeartButtonType, RenderContext } from './types';\n\ninterface Props {\n  embed: ImageMetaData;\n  previewAlt?: boolean;\n  path?: string;\n  heartButton?: HeartButtonType;\n  canonicalUrl?: CanonicalUrlFuncs['image'];\n  inGrid?: boolean;\n  lang?: string;\n  renderContext?: RenderContext;\n}\n\nexport interface Author {\n  name: string;\n  type: string;\n}\n\nexport const getLicenseCredits = (copyright?: {\n  creators?: Author[];\n  rightsholders?: Author[];\n  processors?: Author[];\n}) => {\n  return {\n    creators: copyright?.creators ?? [],\n    rightsholders: copyright?.rightsholders ?? [],\n    processors: copyright?.processors ?? [],\n  };\n};\n\nexport const errorSvgSrc = `data:image/svg+xml;charset=UTF-8,%3Csvg fill='%238A8888' height='400' viewBox='0 0 24 12' width='100%25' xmlns='http://www.w3.org/2000/svg' style='background-color: %23EFF0F2'%3E%3Cpath d='M0 0h24v24H0V0z' fill='none'/%3E%3Cpath transform='scale(0.3) translate(28, 8.5)' d='M11 15h2v2h-2zm0-8h2v6h-2zm.99-5C6.47 2 2 6.48 2 12s4.47 10 9.99 10C17.52 22 22 17.52 22 12S17.52 2 11.99 2zM12 20c-4.42 0-8-3.58-8-8s3.58-8 8-8 8 3.58 8 8-3.58 8-8 8z'/%3E%3C/svg%3E`;\nconst isSmall = (size?: string): size is 'xsmall' | 'small' => size === 'xsmall' || size === 'small';\n\nconst isAlign = (align?: string): align is 'left' | 'right' => align === 'left' || align === 'right';\n\nconst getFigureType = (size?: string, align?: string): FigureType => {\n  if (size && isSmall(size) && align && isAlign(align)) {\n    return `${size}-${align}`;\n  }\n  if (size && isSmall(size) && !align) {\n    return size as FigureType;\n  }\n  if (align && isAlign(align)) {\n    return align;\n  }\n  return 'full';\n};\n\nconst getSizes = (size?: string, align?: string) => {\n  if (align && size === 'full') {\n    return '(min-width: 1024px) 512px, (min-width: 768px) 350px, 100vw';\n  }\n  if (align && size === 'small') {\n    return '(min-width: 1024px) 350px, (min-width: 768px) 180px, 100vw';\n  }\n  if (align && size === 'xsmall') {\n    return '(min-width: 1024px) 180px, (min-width: 768px) 180px, 100vw';\n  }\n  return '(min-width: 1024px) 1024px, 100vw';\n};\n\nexport const getFocalPoint = (data: ImageEmbedData) => {\n  const focalX = parseFloat(data.focalX ?? '');\n  const focalY = parseFloat(data.focalY ?? '');\n  if (!!focalX && !!focalY) {\n    return { x: focalX, y: focalY };\n  }\n  return undefined;\n};\n\nexport const getCrop = (data: ImageEmbedData) => {\n  const lowerRightX = parseFloat(data.lowerRightX ?? '');\n  const lowerRightY = parseFloat(data.lowerRightY ?? '');\n  const upperLeftX = parseFloat(data.upperLeftX ?? '');\n  const upperLeftY = parseFloat(data.upperLeftY ?? '');\n  if (!!lowerRightX && !!lowerRightY && !!upperLeftX && !!upperLeftY) {\n    return {\n      startX: lowerRightX,\n      startY: lowerRightY,\n      endX: upperLeftX,\n      endY: upperLeftY,\n    };\n  }\n  return undefined;\n};\n\nconst expandedSizes = '(min-width: 1024px) 1024px, 100vw';\n\nconst ImageEmbed = ({\n  embed,\n  previewAlt,\n  heartButton: HeartButton,\n  inGrid,\n  path,\n  lang,\n  canonicalUrl,\n  renderContext = 'article',\n}: Props) => {\n  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));\n  const [imageSizes, setImageSizes] = useState<string | undefined>(undefined);\n\n  const parsedDescription = useMemo(() => {\n    if (embed.embedData.caption || renderContext === 'article') {\n      return embed.embedData.caption ? parse(embed.embedData.caption) : undefined;\n    } else if (embed.status === 'success' && embed.data.caption.caption) {\n      return parse(embed.data.caption.caption);\n    }\n  }, [embed, renderContext]);\n\n  if (embed.status === 'error') {\n    const { align, size } = embed.embedData;\n    const figureType = getFigureType(size, align);\n    return <EmbedErrorPlaceholder type={'image'} figureType={figureType} />;\n  }\n\n  const { data, embedData } = embed;\n\n  const altText = embedData.alt || '';\n\n  const figureType = getFigureType(embedData.size, embedData.align);\n  const sizes = getSizes(embedData.size, embedData.align);\n\n  const focalPoint = getFocalPoint(embedData);\n  const crop = getCrop(embedData);\n\n  const isCopyrighted = data.copyright.license.license.toLowerCase() === COPYRIGHTED;\n\n  return (\n    <Figure\n      type={imageSizes ? undefined : figureType}\n      className={imageSizes ? `c-figure--${embedData.align} expanded` : ''}\n    >\n      <ImageWrapper\n        src={!isCopyrighted ? canonicalUrl?.(data) : undefined}\n        crop={crop}\n        size={embedData.size}\n        pagePath={path}\n      >\n        <Image\n          focalPoint={focalPoint}\n          contentType={data.image.contentType}\n          crop={crop}\n          sizes={imageSizes ?? sizes}\n          alt={altText}\n          src={data.image.imageUrl}\n          border={embedData.border}\n          expandButton={\n            <ExpandButton\n              size={embedData.size}\n              expanded={!!imageSizes}\n              bylineHidden={isBylineHidden}\n              onExpand={() => setImageSizes((p) => (p ? undefined : expandedSizes))}\n              onHideByline={() => setIsBylineHidden((p) => !p)}\n            />\n          }\n          lang={lang}\n        />\n      </ImageWrapper>\n      {isBylineHidden || (isSmall(embedData.size) && !imageSizes) ? null : (\n        <EmbedByline\n          type=\"image\"\n          copyright={data.copyright}\n          description={parsedDescription}\n          bottomRounded\n          visibleAlt={previewAlt ? embed.embedData.alt : ''}\n          inGrid={inGrid}\n        >\n          {HeartButton && !isCopyrighted && <HeartButton embed={embed} />}\n        </EmbedByline>\n      )}\n    </Figure>\n  );\n};\n\nconst HiddenSpan = styled.span`\n  ${utils.visuallyHidden};\n`;\n\ninterface ImageWrapperProps {\n  src?: string;\n  children: React.ReactNode;\n  pagePath?: string;\n  crop?: {\n    startX: number;\n    startY: number;\n    endX: number;\n    endY: number;\n  };\n  size?: string;\n}\nconst hideByline = (size?: string): boolean => {\n  return !!size && size.endsWith('-hide-byline');\n};\n\nconst ImageWrapper = ({ src, crop, size, children, pagePath }: ImageWrapperProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size) || hideByline(size) || !src || (pagePath && src.endsWith(pagePath))) {\n    return <>{children}</>;\n  }\n\n  return (\n    <ImageLink src={src} crop={crop}>\n      {children}\n      <HiddenSpan>{t('license.images.itemImage.ariaLabel')}</HiddenSpan>\n    </ImageLink>\n  );\n};\n\ninterface ExpandButtonProps {\n  size?: string;\n  expanded: boolean;\n  bylineHidden: boolean;\n  onExpand: MouseEventHandler<HTMLButtonElement>;\n  onHideByline: MouseEventHandler<HTMLButtonElement>;\n}\n\nconst ExpandButton = ({ size, expanded, bylineHidden, onExpand, onHideByline }: ExpandButtonProps) => {\n  const { t } = useTranslation();\n  if (isSmall(size)) {\n    return (\n      <button\n        type=\"button\"\n        className=\"c-figure__fullscreen-btn\"\n        aria-label={t(`license.images.itemImage.zoom${expanded ? 'Out' : ''}ImageButtonLabel`)}\n        onClick={onExpand}\n      >\n        {expanded ? <ArrowCollapse /> : <ExpandTwoArrows />}\n      </button>\n    );\n  } else if (hideByline(size)) {\n    return (\n      <button\n        type=\"button\"\n        className=\"c-figure__show-byline-btn\"\n        aria-label={t(`license.images.itemImage.${bylineHidden ? 'expandByline' : 'minimizeByline'}`)}\n        onClick={onHideByline}\n      >\n        {bylineHidden ? <ChevronDown /> : <ChevronUp />}\n      </button>\n    );\n  } else return null;\n};\n\nexport default ImageEmbed;\n"]} */"));
200
201
  var hideByline = function hideByline(size) {
201
202
  return !!size && size.endsWith('-hide-byline');
202
203
  };
@@ -19,4 +19,4 @@ export { ConceptNotionV2 } from './conceptComponents';
19
19
  export { default as ConceptListEmbed } from './ConceptListEmbed';
20
20
  export { default as UnknownEmbed } from './UnknownEmbed';
21
21
  export { InlineConcept, BlockConcept } from './ConceptEmbed';
22
- export type { HeartButtonType, RenderContext } from './types';
22
+ export type { HeartButtonType, CanonicalUrlFuncs, RenderContext } from './types';
@@ -7,9 +7,17 @@
7
7
  */
8
8
  import { ElementType } from 'react';
9
9
  import { EmbedMetaData } from '@ndla/types-embed';
10
+ import { IImageMetaInformationV3 } from '@ndla/types-backend/image-api';
10
11
  export type HeartButtonType = ElementType<{
11
12
  embed: Extract<EmbedMetaData, {
12
13
  status: 'success';
13
14
  }>;
14
15
  }>;
16
+ export type EmbedParameter<T extends EmbedMetaData['resource']> = Partial<Extract<EmbedMetaData, {
17
+ status: 'success';
18
+ resource: T;
19
+ }>>;
20
+ export type CanonicalUrlFuncs = {
21
+ image?: (image: IImageMetaInformationV3) => string;
22
+ };
15
23
  export type RenderContext = 'article' | 'embed';
package/lib/index.d.ts CHANGED
@@ -84,7 +84,7 @@ export { BlogPostV2 } from './BlogPost';
84
84
  export { ProgrammeCard } from './ProgrammeCard';
85
85
  export { KeyFigure } from './KeyFigure';
86
86
  export { default as ContactBlock } from './ContactBlock';
87
- export type { HeartButtonType, RenderContext } from './Embed';
87
+ export type { HeartButtonType, CanonicalUrlFuncs, RenderContext } from './Embed';
88
88
  export { CampaignBlock } from './CampaignBlock';
89
89
  export { Grid, GridParallaxItem } from './Grid';
90
90
  export type { GridType } from './Grid';
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ndla/ui",
3
- "version": "49.0.6",
3
+ "version": "49.0.7",
4
4
  "description": "UI component library for NDLA.",
5
5
  "license": "GPL-3.0",
6
6
  "main": "lib/index.js",
@@ -31,23 +31,23 @@
31
31
  "types"
32
32
  ],
33
33
  "dependencies": {
34
- "@ndla/accordion": "^3.0.3",
35
- "@ndla/button": "^12.0.9",
36
- "@ndla/carousel": "^4.0.11",
37
- "@ndla/core": "^4.2.5",
38
- "@ndla/dropdown-menu": "^1.0.14",
39
- "@ndla/forms": "^5.0.11",
34
+ "@ndla/accordion": "^3.0.4",
35
+ "@ndla/button": "^12.0.10",
36
+ "@ndla/carousel": "^4.0.12",
37
+ "@ndla/core": "^4.2.6",
38
+ "@ndla/dropdown-menu": "^1.0.15",
39
+ "@ndla/forms": "^5.0.12",
40
40
  "@ndla/hooks": "^2.1.1",
41
- "@ndla/icons": "^4.1.6",
41
+ "@ndla/icons": "^4.1.7",
42
42
  "@ndla/licenses": "^7.2.2",
43
- "@ndla/modal": "^5.0.8",
44
- "@ndla/notion": "^6.0.11",
45
- "@ndla/safelink": "^4.1.34",
46
- "@ndla/select": "^3.1.6",
47
- "@ndla/switch": "^1.1.20",
48
- "@ndla/tabs": "^3.2.0",
49
- "@ndla/tooltip": "^6.0.2",
50
- "@ndla/typography": "^0.2.7",
43
+ "@ndla/modal": "^5.0.9",
44
+ "@ndla/notion": "^6.0.12",
45
+ "@ndla/safelink": "^4.1.35",
46
+ "@ndla/select": "^3.1.7",
47
+ "@ndla/switch": "^1.1.21",
48
+ "@ndla/tabs": "^3.2.1",
49
+ "@ndla/tooltip": "^6.0.3",
50
+ "@ndla/typography": "^0.3.0",
51
51
  "@ndla/util": "^4.0.0",
52
52
  "@radix-ui/react-popover": "^1.0.7",
53
53
  "@radix-ui/react-radio-group": "^1.1.3",
@@ -80,5 +80,5 @@
80
80
  "publishConfig": {
81
81
  "access": "public"
82
82
  },
83
- "gitHead": "780199a967ead3bcb024102e21fea869e3aec2ba"
83
+ "gitHead": "5b6502dbc509b761ddea72112df15ee4fe633255"
84
84
  }
@@ -6,6 +6,7 @@
6
6
  *
7
7
  */
8
8
 
9
+ import { ReactNode } from 'react';
9
10
  import { useTranslation } from 'react-i18next';
10
11
  import concat from 'lodash/concat';
11
12
  import { css } from '@emotion/react';
@@ -13,10 +14,11 @@ import styled from '@emotion/styled';
13
14
  import { IImageMetaInformationV3 } from '@ndla/types-backend/image-api';
14
15
  import { spacing, fonts, colors, mq, breakpoints, misc } from '@ndla/core';
15
16
  import { BlobPointy, BlobRound } from '@ndla/icons/common';
16
- import { getLicenseByAbbreviation } from '@ndla/licenses';
17
+ import { COPYRIGHTED, getLicenseByAbbreviation } from '@ndla/licenses';
17
18
  import { errorSvgSrc } from '../Embed/ImageEmbed';
18
- import Image from '../Image';
19
+ import Image, { ImageLink } from '../Image';
19
20
  import LicenseLink from '../LicenseByline/LicenseLink';
21
+ import { CanonicalUrlFuncs } from '../Embed';
20
22
 
21
23
  const BLOB_WIDTH = 90;
22
24
 
@@ -31,6 +33,7 @@ interface Props {
31
33
  email: string;
32
34
  embedAlt?: string;
33
35
  lang?: string;
36
+ imageCanonicalUrl?: CanonicalUrlFuncs['image'];
34
37
  }
35
38
  const BlockWrapper = styled.div`
36
39
  display: flex;
@@ -130,6 +133,18 @@ const StyledImage = styled(Image)`
130
133
  object-fit: cover;
131
134
  `;
132
135
 
136
+ interface LinkWrapperProps {
137
+ src?: string;
138
+ children: ReactNode;
139
+ }
140
+
141
+ const LinkWrapper = ({ src, children }: LinkWrapperProps) => {
142
+ if (src) {
143
+ return <ImageLink src={src}>{children}</ImageLink>;
144
+ }
145
+ return children;
146
+ };
147
+
133
148
  const ContactBlock = ({
134
149
  image,
135
150
  jobTitle,
@@ -139,6 +154,7 @@ const ContactBlock = ({
139
154
  embedAlt,
140
155
  blobColor = 'green',
141
156
  blob = 'pointy',
157
+ imageCanonicalUrl,
142
158
  lang,
143
159
  }: Props) => {
144
160
  const { t, i18n } = useTranslation();
@@ -149,16 +165,20 @@ const ContactBlock = ({
149
165
  ? getLicenseByAbbreviation(image.copyright.license.license, i18n.language)
150
166
  : undefined;
151
167
 
168
+ const isCopyrighted = image?.copyright.license.license.toLowerCase() === COPYRIGHTED;
169
+
152
170
  return (
153
171
  <BlockWrapper>
154
172
  <ImageWrapper>
155
173
  {image ? (
156
174
  <>
157
- <StyledImage
158
- alt={embedAlt !== undefined ? embedAlt : image.alttext.alttext}
159
- src={image.image.imageUrl}
160
- sizes={`(min-width: ${breakpoints.tablet}) 240px, (max-width: ${breakpoints.tablet}) 500px`}
161
- />
175
+ <LinkWrapper src={!isCopyrighted && image ? imageCanonicalUrl?.(image) : undefined}>
176
+ <StyledImage
177
+ alt={embedAlt !== undefined ? embedAlt : image.alttext.alttext}
178
+ src={image.image.imageUrl}
179
+ sizes={`(min-width: ${breakpoints.tablet}) 240px, (max-width: ${breakpoints.tablet}) 500px`}
180
+ />
181
+ </LinkWrapper>
162
182
  <span>
163
183
  {`${t('photo')}: ${authors.reduce((acc, name) => (acc = `${acc} ${name?.name}`), '')} `}
164
184
  {!!license && <LicenseLink license={license} asLink={!!license.url.length} />}
@@ -19,13 +19,14 @@ import { Figure, FigureType } from '../Figure';
19
19
  import Image, { ImageLink } from '../Image';
20
20
  import { EmbedByline } from '../LicenseByline';
21
21
  import EmbedErrorPlaceholder from './EmbedErrorPlaceholder';
22
- import { HeartButtonType, RenderContext } from './types';
22
+ import { CanonicalUrlFuncs, HeartButtonType, RenderContext } from './types';
23
23
 
24
24
  interface Props {
25
25
  embed: ImageMetaData;
26
26
  previewAlt?: boolean;
27
27
  path?: string;
28
28
  heartButton?: HeartButtonType;
29
+ canonicalUrl?: CanonicalUrlFuncs['image'];
29
30
  inGrid?: boolean;
30
31
  lang?: string;
31
32
  renderContext?: RenderContext;
@@ -113,6 +114,7 @@ const ImageEmbed = ({
113
114
  inGrid,
114
115
  path,
115
116
  lang,
117
+ canonicalUrl,
116
118
  renderContext = 'article',
117
119
  }: Props) => {
118
120
  const [isBylineHidden, setIsBylineHidden] = useState(hideByline(embed.embedData.size));
@@ -150,7 +152,7 @@ const ImageEmbed = ({
150
152
  className={imageSizes ? `c-figure--${embedData.align} expanded` : ''}
151
153
  >
152
154
  <ImageWrapper
153
- src={!isCopyrighted ? embedData.pageUrl || data.image.imageUrl : undefined}
155
+ src={!isCopyrighted ? canonicalUrl?.(data) : undefined}
154
156
  crop={crop}
155
157
  size={embedData.size}
156
158
  pagePath={path}
@@ -20,4 +20,4 @@ export { ConceptNotionV2 } from './conceptComponents';
20
20
  export { default as ConceptListEmbed } from './ConceptListEmbed';
21
21
  export { default as UnknownEmbed } from './UnknownEmbed';
22
22
  export { InlineConcept, BlockConcept } from './ConceptEmbed';
23
- export type { HeartButtonType, RenderContext } from './types';
23
+ export type { HeartButtonType, CanonicalUrlFuncs, RenderContext } from './types';
@@ -8,7 +8,16 @@
8
8
 
9
9
  import { ElementType } from 'react';
10
10
  import { EmbedMetaData } from '@ndla/types-embed';
11
+ import { IImageMetaInformationV3 } from '@ndla/types-backend/image-api';
11
12
 
12
13
  export type HeartButtonType = ElementType<{ embed: Extract<EmbedMetaData, { status: 'success' }> }>;
13
14
 
15
+ export type EmbedParameter<T extends EmbedMetaData['resource']> = Partial<
16
+ Extract<EmbedMetaData, { status: 'success'; resource: T }>
17
+ >;
18
+
19
+ export type CanonicalUrlFuncs = {
20
+ image?: (image: IImageMetaInformationV3) => string;
21
+ };
22
+
14
23
  export type RenderContext = 'article' | 'embed';
package/src/index.ts CHANGED
@@ -213,7 +213,7 @@ export { BlogPostV2 } from './BlogPost';
213
213
  export { ProgrammeCard } from './ProgrammeCard';
214
214
  export { KeyFigure } from './KeyFigure';
215
215
  export { default as ContactBlock } from './ContactBlock';
216
- export type { HeartButtonType, RenderContext } from './Embed';
216
+ export type { HeartButtonType, CanonicalUrlFuncs, RenderContext } from './Embed';
217
217
  export { CampaignBlock } from './CampaignBlock';
218
218
  export { Grid, GridParallaxItem } from './Grid';
219
219
  export type { GridType } from './Grid';