@ndla/ui 27.1.7 → 29.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (111) hide show
  1. package/es/Frontpage/FrontpageAllSubjects.js +9 -8
  2. package/es/Resource/BlockResource.js +22 -13
  3. package/es/Resource/ListResource.js +24 -15
  4. package/es/Resource/resourceComponents.js +28 -28
  5. package/es/TagSelector/Control.js +23 -0
  6. package/es/TagSelector/DropdownIndicator.js +66 -0
  7. package/es/TagSelector/Input.js +19 -0
  8. package/es/TagSelector/Menu.js +26 -0
  9. package/es/TagSelector/MenuList.js +22 -0
  10. package/es/TagSelector/Option.js +55 -0
  11. package/es/TagSelector/SelectContainer.js +18 -0
  12. package/es/TagSelector/TagSelector.js +161 -100
  13. package/es/TagSelector/ValueButton.js +46 -0
  14. package/es/TagSelector/ariaMessages.js +104 -0
  15. package/es/TagSelector/index.js +2 -1
  16. package/es/TagSelector/types.js +0 -0
  17. package/es/TreeStructure/ComboboxButton.js +19 -18
  18. package/es/TreeStructure/TreeStructure.js +8 -8
  19. package/es/locale/messages-en.js +46 -8
  20. package/es/locale/messages-nb.js +57 -19
  21. package/es/locale/messages-nn.js +56 -18
  22. package/es/locale/messages-se.js +48 -10
  23. package/es/locale/messages-sma.js +57 -19
  24. package/es/model/ContentType.js +23 -1
  25. package/es/model/SubjectCategories.js +1 -5
  26. package/es/model/index.js +3 -2
  27. package/lib/Frontpage/FrontpageAllSubjects.js +10 -8
  28. package/lib/Resource/BlockResource.d.ts +6 -3
  29. package/lib/Resource/BlockResource.js +23 -12
  30. package/lib/Resource/ListResource.d.ts +6 -3
  31. package/lib/Resource/ListResource.js +25 -14
  32. package/lib/Resource/resourceComponents.d.ts +6 -3
  33. package/lib/Resource/resourceComponents.js +30 -30
  34. package/lib/TagSelector/Control.d.ts +12 -0
  35. package/lib/TagSelector/Control.js +35 -0
  36. package/lib/TagSelector/DropdownIndicator.d.ts +12 -0
  37. package/lib/TagSelector/DropdownIndicator.js +80 -0
  38. package/lib/TagSelector/Input.d.ts +12 -0
  39. package/lib/TagSelector/Input.js +33 -0
  40. package/lib/TagSelector/Menu.d.ts +12 -0
  41. package/lib/TagSelector/Menu.js +40 -0
  42. package/lib/TagSelector/MenuList.d.ts +13 -0
  43. package/lib/TagSelector/MenuList.js +36 -0
  44. package/lib/TagSelector/Option.d.ts +12 -0
  45. package/lib/TagSelector/Option.js +61 -0
  46. package/lib/TagSelector/SelectContainer.d.ts +12 -0
  47. package/lib/TagSelector/SelectContainer.js +31 -0
  48. package/lib/TagSelector/TagSelector.d.ts +14 -11
  49. package/lib/TagSelector/TagSelector.js +165 -96
  50. package/lib/TagSelector/ValueButton.d.ts +16 -0
  51. package/lib/TagSelector/ValueButton.js +55 -0
  52. package/lib/TagSelector/ariaMessages.d.ts +16 -0
  53. package/lib/TagSelector/ariaMessages.js +113 -0
  54. package/lib/TagSelector/index.d.ts +2 -1
  55. package/lib/TagSelector/index.js +3 -5
  56. package/lib/TagSelector/types.d.ts +11 -0
  57. package/lib/TagSelector/types.js +1 -0
  58. package/lib/TreeStructure/ComboboxButton.js +19 -18
  59. package/lib/TreeStructure/TreeStructure.js +7 -7
  60. package/lib/locale/messages-en.d.ts +56 -25
  61. package/lib/locale/messages-en.js +46 -8
  62. package/lib/locale/messages-nb.d.ts +56 -25
  63. package/lib/locale/messages-nb.js +57 -19
  64. package/lib/locale/messages-nn.d.ts +56 -25
  65. package/lib/locale/messages-nn.js +56 -18
  66. package/lib/locale/messages-se.d.ts +56 -25
  67. package/lib/locale/messages-se.js +48 -10
  68. package/lib/locale/messages-sma.d.ts +56 -25
  69. package/lib/locale/messages-sma.js +57 -19
  70. package/lib/model/ContentType.d.ts +18 -0
  71. package/lib/model/ContentType.js +32 -2
  72. package/lib/model/SubjectCategories.d.ts +0 -3
  73. package/lib/model/SubjectCategories.js +3 -10
  74. package/lib/model/index.d.ts +12 -2
  75. package/lib/model/index.js +4 -3
  76. package/package.json +15 -14
  77. package/src/Frontpage/FrontpageAllSubjects.tsx +5 -2
  78. package/src/Resource/BlockResource.tsx +18 -11
  79. package/src/Resource/ListResource.tsx +14 -11
  80. package/src/Resource/resourceComponents.tsx +13 -14
  81. package/src/TagSelector/Control.tsx +34 -0
  82. package/src/TagSelector/DropdownIndicator.tsx +47 -0
  83. package/src/TagSelector/Input.tsx +31 -0
  84. package/src/TagSelector/Menu.tsx +38 -0
  85. package/src/TagSelector/MenuList.tsx +30 -0
  86. package/src/TagSelector/Option.tsx +53 -0
  87. package/src/TagSelector/SelectContainer.tsx +32 -0
  88. package/src/TagSelector/TagSelector.tsx +105 -84
  89. package/src/TagSelector/ValueButton.tsx +46 -0
  90. package/src/TagSelector/ariaMessages.ts +87 -0
  91. package/src/TagSelector/index.ts +2 -1
  92. package/src/TagSelector/types.ts +12 -0
  93. package/src/TreeStructure/ComboboxButton.tsx +15 -17
  94. package/src/TreeStructure/TreeStructure.tsx +2 -11
  95. package/src/locale/messages-en.ts +46 -9
  96. package/src/locale/messages-nb.ts +57 -20
  97. package/src/locale/messages-nn.ts +56 -19
  98. package/src/locale/messages-se.ts +48 -11
  99. package/src/locale/messages-sma.ts +57 -20
  100. package/src/model/ContentType.ts +29 -0
  101. package/src/model/SubjectCategories.ts +0 -5
  102. package/src/model/index.ts +2 -1
  103. package/es/TagSelector/SuggestionInput.js +0 -285
  104. package/es/TagSelector/Suggestions.js +0 -97
  105. package/lib/TagSelector/SuggestionInput.d.ts +0 -19
  106. package/lib/TagSelector/SuggestionInput.js +0 -299
  107. package/lib/TagSelector/Suggestions.d.ts +0 -12
  108. package/lib/TagSelector/Suggestions.js +0 -99
  109. package/src/.DS_Store +0 -0
  110. package/src/TagSelector/SuggestionInput.tsx +0 -287
  111. package/src/TagSelector/Suggestions.tsx +0 -139
@@ -23,6 +23,8 @@ var _ContentLoader = _interopRequireDefault(require("../ContentLoader"));
23
23
 
24
24
  var _ContentTypeBadge = _interopRequireDefault(require("../ContentTypeBadge"));
25
25
 
26
+ var _ContentType = require("../model/ContentType");
27
+
26
28
  var _core2 = require("@emotion/core");
27
29
 
28
30
  function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; }
@@ -36,7 +38,7 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
36
38
  var StyledResourceDescription = (0, _styledBase["default"])("p", {
37
39
  target: "e18z2zfy0",
38
40
  label: "StyledResourceDescription"
39
- })("grid-area:description;line-clamp:2;line-height:1em;height:3.1em;margin:0;overflow:hidden;", _core.fonts.sizes(16), ";text-overflow:ellipsis;display:-webkit-box;-webkit-line-clamp:2;line-clamp:2;-webkit-box-orient:vertical;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAyB0C","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  TopicList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  topics: string[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n  contentType: string;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst TopicAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  topics,\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n  contentType,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentType}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <TopicAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <ResourceTitle>{title}</ResourceTitle>\n          </ResourceTitleLink>\n          <TopicList topics={topics} />\n        </TopicAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
41
+ })("grid-area:description;line-clamp:2;line-height:1em;height:3.1em;margin:0;overflow:hidden;", _core.fonts.sizes(16), ";text-overflow:ellipsis;display:-webkit-box;-webkit-line-clamp:2;line-clamp:2;-webkit-box-orient:vertical;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA0B0C","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  headingLevel?: 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6';\n  resourceTypes: { id: string; name: string }[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst ResourceTypeAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  resourceTypes,\n  headingLevel = 'h2',\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <ResourceTypeAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <Title>{title}</Title>\n          </ResourceTitleLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </ResourceTypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
40
42
  var ResourceWrapper = (0, _styledBase["default"])("div", {
41
43
  target: "e18z2zfy1",
42
44
  label: "ResourceWrapper"
@@ -44,13 +46,13 @@ var ResourceWrapper = (0, _styledBase["default"])("div", {
44
46
  until: _core.breakpoints.mobileWide
45
47
  }), "{grid-template-columns:auto 1fr;grid-template-areas:'image topicAndTitle' 'description description' 'tags tags';}cursor:pointer;padding:", _core.spacing.small, ";border:1px solid ", _core.colors.brand.neutral7, ";border-radius:2px;gap:0 ", _core.spacing.small, ";&:hover{box-shadow:1px 1px 6px 2px rgba(9,55,101,0.08);transition-duration:0.2s;", function () {
46
48
  return _resourceComponents.ResourceTitleLink;
47
- }, "{color:", _core.colors.brand.primary, ";text-decoration:underline;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAyCkC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  TopicList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  topics: string[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n  contentType: string;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst TopicAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  topics,\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n  contentType,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentType}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <TopicAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <ResourceTitle>{title}</ResourceTitle>\n          </ResourceTitleLink>\n          <TopicList topics={topics} />\n        </TopicAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
49
+ }, "{color:", _core.colors.brand.primary, ";text-decoration:underline;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA0CkC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  headingLevel?: 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6';\n  resourceTypes: { id: string; name: string }[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst ResourceTypeAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  resourceTypes,\n  headingLevel = 'h2',\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <ResourceTypeAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <Title>{title}</Title>\n          </ResourceTitleLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </ResourceTypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
48
50
  var TagsandActionMenu = (0, _styledBase["default"])("div", {
49
51
  target: "e18z2zfy2",
50
52
  label: "TagsandActionMenu"
51
53
  })("box-sizing:content-box;padding:2px;grid-area:tags;display:flex;align-items:center;width:100%;overflow:hidden;align-self:flex-start;justify-self:flex-end;justify-content:flex-end;", _core.mq.range({
52
54
  from: _core.breakpoints.mobileWide
53
- }), "{margin-top:-", _core.spacing.xsmall, ";margin-right:-", _core.spacing.xxsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAyEoC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  TopicList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  topics: string[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n  contentType: string;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst TopicAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  topics,\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n  contentType,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentType}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <TopicAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <ResourceTitle>{title}</ResourceTitle>\n          </ResourceTitleLink>\n          <TopicList topics={topics} />\n        </TopicAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
55
+ }), "{margin-top:-", _core.spacing.xsmall, ";margin-right:-", _core.spacing.xxsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA0EoC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  headingLevel?: 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6';\n  resourceTypes: { id: string; name: string }[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst ResourceTypeAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  resourceTypes,\n  headingLevel = 'h2',\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <ResourceTypeAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <Title>{title}</Title>\n          </ResourceTitleLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </ResourceTypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
54
56
  var StyledImageWrapper = (0, _styledBase["default"])("div", {
55
57
  target: "e18z2zfy3",
56
58
  label: "StyledImageWrapper"
@@ -58,7 +60,7 @@ var StyledImageWrapper = (0, _styledBase["default"])("div", {
58
60
  return p.imageSize === 'normal' ? '136px' : '56px';
59
61
  }, ";", _core.mq.range({
60
62
  until: _core.breakpoints.mobileWide
61
- }), "{width:54px;height:40px;}overflow:hidden;border-radius:2px;display:flex;align-items:center;justify-content:center;aspect-ratio:4/3;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA2FuD","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  TopicList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  topics: string[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n  contentType: string;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst TopicAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  topics,\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n  contentType,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentType}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <TopicAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <ResourceTitle>{title}</ResourceTitle>\n          </ResourceTitleLink>\n          <TopicList topics={topics} />\n        </TopicAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
63
+ }), "{width:54px;height:40px;}overflow:hidden;border-radius:2px;display:flex;align-items:center;justify-content:center;aspect-ratio:4/3;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA4FuD","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  headingLevel?: 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6';\n  resourceTypes: { id: string; name: string }[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst ResourceTypeAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  resourceTypes,\n  headingLevel = 'h2',\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <ResourceTypeAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <Title>{title}</Title>\n          </ResourceTitleLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </ResourceTypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
62
64
  var StyledImage = ( /*#__PURE__*/0, _styledBase["default"])(_Image["default"], {
63
65
  target: "e18z2zfy4",
64
66
  label: "StyledImage"
@@ -68,7 +70,7 @@ var StyledImage = ( /*#__PURE__*/0, _styledBase["default"])(_Image["default"], {
68
70
  } : {
69
71
  name: "1t32sn2",
70
72
  styles: "object-fit:cover;aspect-ratio:4/3;",
71
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA0GiC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  TopicList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  topics: string[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n  contentType: string;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst TopicAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  topics,\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n  contentType,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentType}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <TopicAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <ResourceTitle>{title}</ResourceTitle>\n          </ResourceTitleLink>\n          <TopicList topics={topics} />\n        </TopicAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */",
73
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA2GiC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  headingLevel?: 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6';\n  resourceTypes: { id: string; name: string }[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst ResourceTypeAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  resourceTypes,\n  headingLevel = 'h2',\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <ResourceTypeAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <Title>{title}</Title>\n          </ResourceTitleLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </ResourceTypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */",
72
74
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
73
75
  });
74
76
  var TopicAndTitleWrapper = (0, _styledBase["default"])("div", {
@@ -80,7 +82,7 @@ var TopicAndTitleWrapper = (0, _styledBase["default"])("div", {
80
82
  } : {
81
83
  name: "2x4or9",
82
84
  styles: "grid-area:topicAndTitle;margin-top:2px;",
83
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA+GuC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  TopicList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  topics: string[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n  contentType: string;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst TopicAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  topics,\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n  contentType,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentType}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <TopicAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <ResourceTitle>{title}</ResourceTitle>\n          </ResourceTitleLink>\n          <TopicList topics={topics} />\n        </TopicAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */",
85
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAgHuC","file":"ListResource.tsx","sourcesContent":["/**\n * Copyright (c) 2022-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 styled from '@emotion/styled';\nimport React, { useRef } from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuButton, MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0;\n  overflow: hidden;\n  ${fonts.sizes(16)};\n  text-overflow: ellipsis;\n  // Unfortunate css needed for multi-line text overflow ellipsis.\n  display: -webkit-box;\n  -webkit-line-clamp: 2;\n  line-clamp: 2;\n  -webkit-box-orient: vertical;\n`;\n\nconst ResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\n\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    grid-template-columns: auto 1fr;\n    grid-template-areas:\n      'image                topicAndTitle'\n      'description          description'\n      'tags                 tags';\n  }\n\n  cursor: pointer;\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n  gap: 0 ${spacing.small};\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => ResourceTitleLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\nconst TagsandActionMenu = styled.div`\n  box-sizing: content-box;\n  padding: 2px;\n  grid-area: tags;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  overflow: hidden;\n  align-self: flex-start;\n  justify-self: flex-end;\n  justify-content: flex-end;\n\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin-top: -${spacing.xsmall};\n    margin-right: -${spacing.xxsmall};\n  }\n`;\n\nconst StyledImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 54px;\n    height: 40px;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  aspect-ratio: 4/3;\n`;\n\nconst StyledImage = styled(Image)`\n  object-fit: cover;\n  aspect-ratio: 4/3;\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  margin-top: 2px;\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nexport interface ListResourceProps {\n  id: string;\n  link: string;\n  tagLinkPrefix?: string;\n  title: string;\n  resourceImage: ResourceImageProps;\n  headingLevel?: 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6';\n  resourceTypes: { id: string; name: string }[];\n  tags?: string[];\n  description?: string;\n  menuItems?: MenuItemProps[];\n  isLoading?: boolean;\n  targetBlank?: boolean;\n}\n\ninterface ListResourceImageProps {\n  resourceImage: ResourceImageProps;\n  loading?: boolean;\n  type: 'normal' | 'compact';\n  contentType: string;\n}\n\nconst ListResourceImage = ({ resourceImage, loading, type, contentType }: ListResourceImageProps) => {\n  if (!loading) {\n    if (resourceImage.src === '') {\n      return (\n        <StyledContentIconWrapper contentType={contentType}>\n          <ContentTypeBadge type={contentType} size=\"x-small\" />\n        </StyledContentIconWrapper>\n      );\n    } else {\n      return (\n        <StyledImage alt={resourceImage.alt} src={resourceImage.src} fallbackWidth={type === 'compact' ? 56 : 136} />\n      );\n    }\n  }\n\n  return (\n    <ContentLoader height={'100%'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n      <rect\n        x=\"0\"\n        y=\"0\"\n        rx=\"3\"\n        ry=\"3\"\n        width={type === 'compact' ? '56' : '136'}\n        height={type === 'compact' ? '40' : '96'}\n      />\n    </ContentLoader>\n  );\n};\n\nconst ResourceTypeAndTitleLoader = ({ loading, children }: LoaderProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'40px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" rx=\"3\" ry=\"3\" width={'100%'} height={'16'} />\n        <rect x=\"0\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n        <rect x=\"80\" y=\"18\" rx=\"3\" ry=\"3\" width={'70'} height={'16'} />\n      </ContentLoader>\n    );\n  }\n  return <>{children}</>;\n};\n\ninterface ResourceDescriptionProps {\n  description?: string;\n  loading?: boolean;\n}\n\nconst ResourceDescription = ({ description, loading }: ResourceDescriptionProps) => {\n  if (loading) {\n    return (\n      <ContentLoader height={'20px'} width={'100%'} viewBox={null} preserveAspectRatio=\"none\">\n        <rect x=\"0\" y=\"0\" width=\"100%\" height=\"20\" />\n      </ContentLoader>\n    );\n  }\n  return <StyledResourceDescription>{description}</StyledResourceDescription>;\n};\n\nconst ListResource = ({\n  id,\n  link,\n  tagLinkPrefix,\n  title,\n  tags,\n  resourceImage,\n  resourceTypes,\n  headingLevel = 'h2',\n  description,\n  menuItems,\n  isLoading = false,\n  targetBlank,\n}: ListResourceProps) => {\n  const showDescription = description !== undefined;\n  const imageType = showDescription ? 'normal' : 'compact';\n  const linkRef = useRef<HTMLAnchorElement>(null);\n  const firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n  const handleClick = () => {\n    if (linkRef.current) {\n      linkRef.current.click();\n    }\n  };\n\n  return (\n    <ResourceWrapper onClick={handleClick} id={id}>\n      <StyledImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </StyledImageWrapper>\n      <TopicAndTitleWrapper>\n        <ResourceTypeAndTitleLoader loading={isLoading}>\n          <ResourceTitleLink to={link} target={targetBlank ? '_blank' : undefined} ref={linkRef}>\n            <Title>{title}</Title>\n          </ResourceTitleLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </ResourceTypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <ResourceDescription description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton alignRight size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */",
84
86
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
85
87
  });
86
88
 
@@ -122,7 +124,7 @@ var ListResourceImage = function ListResourceImage(_ref) {
122
124
  }));
123
125
  };
124
126
 
125
- var TopicAndTitleLoader = function TopicAndTitleLoader(_ref2) {
127
+ var ResourceTypeAndTitleLoader = function ResourceTypeAndTitleLoader(_ref2) {
126
128
  var loading = _ref2.loading,
127
129
  children = _ref2.children;
128
130
 
@@ -181,22 +183,31 @@ var ResourceDescription = function ResourceDescription(_ref3) {
181
183
  };
182
184
 
183
185
  var ListResource = function ListResource(_ref4) {
186
+ var _resourceTypes$0$id, _resourceTypes$, _contentTypeMapping$f;
187
+
184
188
  var id = _ref4.id,
185
189
  link = _ref4.link,
186
190
  tagLinkPrefix = _ref4.tagLinkPrefix,
187
191
  title = _ref4.title,
188
192
  tags = _ref4.tags,
189
193
  resourceImage = _ref4.resourceImage,
190
- topics = _ref4.topics,
194
+ resourceTypes = _ref4.resourceTypes,
195
+ _ref4$headingLevel = _ref4.headingLevel,
196
+ headingLevel = _ref4$headingLevel === void 0 ? 'h2' : _ref4$headingLevel,
191
197
  description = _ref4.description,
192
198
  menuItems = _ref4.menuItems,
193
199
  _ref4$isLoading = _ref4.isLoading,
194
200
  isLoading = _ref4$isLoading === void 0 ? false : _ref4$isLoading,
195
- targetBlank = _ref4.targetBlank,
196
- contentType = _ref4.contentType;
201
+ targetBlank = _ref4.targetBlank;
197
202
  var showDescription = description !== undefined;
198
203
  var imageType = showDescription ? 'normal' : 'compact';
199
204
  var linkRef = (0, _react.useRef)(null);
205
+ var firstContentType = (_resourceTypes$0$id = resourceTypes === null || resourceTypes === void 0 ? void 0 : (_resourceTypes$ = resourceTypes[0]) === null || _resourceTypes$ === void 0 ? void 0 : _resourceTypes$.id) !== null && _resourceTypes$0$id !== void 0 ? _resourceTypes$0$id : '';
206
+
207
+ var Title = _resourceComponents.ResourceTitle.withComponent(headingLevel, {
208
+ target: "e18z2zfy6",
209
+ label: "Title"
210
+ });
200
211
 
201
212
  var handleClick = function handleClick() {
202
213
  if (linkRef.current) {
@@ -213,15 +224,15 @@ var ListResource = function ListResource(_ref4) {
213
224
  resourceImage: resourceImage,
214
225
  loading: isLoading,
215
226
  type: imageType,
216
- contentType: contentType
217
- })), (0, _core2.jsx)(TopicAndTitleWrapper, null, (0, _core2.jsx)(TopicAndTitleLoader, {
227
+ contentType: (_contentTypeMapping$f = _ContentType.contentTypeMapping[firstContentType]) !== null && _contentTypeMapping$f !== void 0 ? _contentTypeMapping$f : _ContentType.contentTypeMapping['default']
228
+ })), (0, _core2.jsx)(TopicAndTitleWrapper, null, (0, _core2.jsx)(ResourceTypeAndTitleLoader, {
218
229
  loading: isLoading
219
230
  }, (0, _core2.jsx)(_resourceComponents.ResourceTitleLink, {
220
231
  to: link,
221
232
  target: targetBlank ? '_blank' : undefined,
222
233
  ref: linkRef
223
- }, (0, _core2.jsx)(_resourceComponents.ResourceTitle, null, title)), (0, _core2.jsx)(_resourceComponents.TopicList, {
224
- topics: topics
234
+ }, (0, _core2.jsx)(Title, null, title)), (0, _core2.jsx)(_resourceComponents.ResourceTypeList, {
235
+ resourceTypes: resourceTypes
225
236
  }))), showDescription && (0, _core2.jsx)(ResourceDescription, {
226
237
  description: description,
227
238
  loading: isLoading
@@ -31,8 +31,11 @@ interface CompressedTagListProps {
31
31
  tagLinkPrefix?: string;
32
32
  }
33
33
  export declare const CompressedTagList: ({ tags, tagLinkPrefix }: CompressedTagListProps) => JSX.Element;
34
- interface TopicListProps {
35
- topics?: string[];
34
+ interface ResourceTypeListProps {
35
+ resourceTypes?: {
36
+ id: string;
37
+ name: string;
38
+ }[];
36
39
  }
37
- export declare const TopicList: ({ topics }: TopicListProps) => JSX.Element | null;
40
+ export declare const ResourceTypeList: ({ resourceTypes }: ResourceTypeListProps) => JSX.Element | null;
38
41
  export {};