@ndla/ui 35.0.15 → 35.0.17

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 (48) hide show
  1. package/README.md +1 -1
  2. package/es/MyNdla/Resource/Folder.js +17 -16
  3. package/es/MyNdla/Resource/FolderMenu.js +74 -0
  4. package/es/MyNdla/SettingsMenu.js +98 -0
  5. package/es/MyNdla/index.js +3 -1
  6. package/es/Resource/BlockResource.js +8 -10
  7. package/es/Resource/ListResource.js +8 -10
  8. package/es/index.js +1 -1
  9. package/es/locale/messages-en.js +3 -1
  10. package/es/locale/messages-nb.js +3 -1
  11. package/es/locale/messages-nn.js +3 -1
  12. package/es/locale/messages-se.js +3 -1
  13. package/es/locale/messages-sma.js +3 -1
  14. package/lib/MyNdla/Resource/Folder.d.ts +4 -3
  15. package/lib/MyNdla/Resource/Folder.js +17 -16
  16. package/lib/MyNdla/Resource/FolderMenu.d.ts +16 -0
  17. package/lib/MyNdla/Resource/FolderMenu.js +81 -0
  18. package/lib/MyNdla/SettingsMenu.d.ts +15 -0
  19. package/lib/MyNdla/SettingsMenu.js +96 -0
  20. package/lib/MyNdla/index.d.ts +3 -1
  21. package/lib/MyNdla/index.js +14 -0
  22. package/lib/Resource/BlockResource.js +8 -10
  23. package/lib/Resource/ListResource.js +8 -10
  24. package/lib/index.d.ts +1 -1
  25. package/lib/index.js +6 -0
  26. package/lib/locale/messages-en.d.ts +2 -0
  27. package/lib/locale/messages-en.js +3 -1
  28. package/lib/locale/messages-nb.d.ts +2 -0
  29. package/lib/locale/messages-nb.js +3 -1
  30. package/lib/locale/messages-nn.d.ts +2 -0
  31. package/lib/locale/messages-nn.js +3 -1
  32. package/lib/locale/messages-se.d.ts +2 -0
  33. package/lib/locale/messages-se.js +3 -1
  34. package/lib/locale/messages-sma.d.ts +2 -0
  35. package/lib/locale/messages-sma.js +3 -1
  36. package/package.json +6 -6
  37. package/src/MyNdla/Resource/Folder.tsx +20 -6
  38. package/src/MyNdla/Resource/FolderMenu.tsx +102 -0
  39. package/src/MyNdla/SettingsMenu.tsx +97 -0
  40. package/src/MyNdla/index.ts +3 -1
  41. package/src/Resource/BlockResource.tsx +3 -2
  42. package/src/Resource/ListResource.tsx +4 -3
  43. package/src/index.ts +1 -1
  44. package/src/locale/messages-en.ts +3 -1
  45. package/src/locale/messages-nb.ts +3 -1
  46. package/src/locale/messages-nn.ts +3 -1
  47. package/src/locale/messages-se.ts +3 -1
  48. package/src/locale/messages-sma.ts +3 -1
@@ -10,12 +10,12 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
10
10
 
11
11
  import React from 'react';
12
12
  import { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';
13
- import { MenuButton } from '@ndla/button';
14
13
  import Image from '../Image';
15
14
  import { CompressedTagList, ResourceTitle, ResourceTitleLink as StyledLink, ResourceTypeList, StyledContentIconWrapper } from './resourceComponents';
16
15
  import ContentLoader from '../ContentLoader';
17
16
  import ContentTypeBadge from '../ContentTypeBadge';
18
17
  import { contentTypeMapping } from '../model/ContentType';
18
+ import { SettingsMenu } from '../MyNdla';
19
19
  import { jsx as _jsx } from "@emotion/react/jsx-runtime";
20
20
  import { jsxs as _jsxs } from "@emotion/react/jsx-runtime";
21
21
  import { Fragment as _Fragment } from "@emotion/react/jsx-runtime";
@@ -26,7 +26,7 @@ var ListResourceWrapper = /*#__PURE__*/_styled("div", {
26
26
  until: breakpoints.mobileWide
27
27
  }), "{grid-template-columns:auto 1fr;grid-template-areas:'image topicAndTitle' 'description description' 'tags tags';}cursor:pointer;border:1px solid ", colors.brand.neutral7, ";border-radius:2px;&:hover{box-shadow:1px 1px 6px 2px rgba(9, 55, 101, 0.08);transition-duration:0.2s;", function () {
28
28
  return StyledLink;
29
- }, "{color:", 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":"AA0BsC","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 as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton align=\"end\" size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
29
+ }, "{color:", 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":"AA2BsC","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 from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\nimport { SettingsMenu } from '../MyNdla';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <SettingsMenu menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
30
30
  var ImageWrapper = /*#__PURE__*/_styled("div", {
31
31
  target: "e18z2zfy4",
32
32
  label: "ImageWrapper"
@@ -34,7 +34,7 @@ var ImageWrapper = /*#__PURE__*/_styled("div", {
34
34
  return p.imageSize === 'normal' ? '136px' : '56px';
35
35
  }, ";", mq.range({
36
36
  until: breakpoints.mobileWide
37
- }), "{width:56px;margin-bottom:0;}overflow:hidden;border-radius:2px;display:flex;margin:", spacing.small, ";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":"AA4DiD","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 as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton align=\"end\" size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
37
+ }), "{width:56px;margin-bottom:0;}overflow:hidden;border-radius:2px;display:flex;margin:", spacing.small, ";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":"AA6DiD","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 from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\nimport { SettingsMenu } from '../MyNdla';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <SettingsMenu menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
38
38
  var StyledImage = /*#__PURE__*/_styled(Image, {
39
39
  target: "e18z2zfy3",
40
40
  label: "StyledImage"
@@ -44,7 +44,7 @@ var StyledImage = /*#__PURE__*/_styled(Image, {
44
44
  } : {
45
45
  name: "ie7djk",
46
46
  styles: "object-fit:cover;aspect-ratio:4/3",
47
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA4EiC","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 as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton align=\"end\" size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */",
47
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AA6EiC","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 from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\nimport { SettingsMenu } from '../MyNdla';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <SettingsMenu menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */",
48
48
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
49
49
  });
50
50
  var StyledResourceDescription = /*#__PURE__*/_styled("p", {
@@ -52,19 +52,19 @@ var StyledResourceDescription = /*#__PURE__*/_styled("p", {
52
52
  label: "StyledResourceDescription"
53
53
  })("grid-area:description;line-clamp:2;line-height:1em;height:3.1em;margin:0 ", spacing.small, " ", spacing.small, " 0;", mq.range({
54
54
  until: breakpoints.mobileWide
55
- }), "{margin:0 ", spacing.small, ";}overflow:hidden;", 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":"AAiF0C","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 as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton align=\"end\" size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
55
+ }), "{margin:0 ", spacing.small, ";}overflow:hidden;", 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":"AAkF0C","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 from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\nimport { SettingsMenu } from '../MyNdla';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <SettingsMenu menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
56
56
  var TagsandActionMenu = /*#__PURE__*/_styled("div", {
57
57
  target: "e18z2zfy1",
58
58
  label: "TagsandActionMenu"
59
59
  })("grid-area:tags;z-index:1;box-sizing:content-box;display:grid;grid-template-columns:1fr auto auto;align-items:center;align-self:flex-start;justify-items:flex-end;overflow:hidden;", mq.range({
60
60
  until: breakpoints.mobileWide
61
- }), "{min-height:", spacing.small, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAoGoC","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 as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton align=\"end\" size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
61
+ }), "{min-height:", spacing.small, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAqGoC","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 from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\nimport { SettingsMenu } from '../MyNdla';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <SettingsMenu menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
62
62
  var TopicAndTitleWrapper = /*#__PURE__*/_styled("div", {
63
63
  target: "e18z2zfy0",
64
64
  label: "TopicAndTitleWrapper"
65
65
  })("grid-area:topicAndTitle;display:flex;margin:", spacing.small, " 0;flex-direction:column;overflow:hidden;margin-right:", spacing.small, ";", mq.range({
66
66
  until: breakpoints.mobileWide
67
- }), "{margin-bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAmHuC","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 as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <MenuButton align=\"end\" size=\"small\" menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
67
+ }), "{margin-bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["ListResource.tsx"],"names":[],"mappings":"AAoHuC","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 from 'react';\nimport { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';\nimport { MenuItemProps } from '@ndla/button';\nimport Image from '../Image';\nimport {\n  CompressedTagList,\n  ResourceImageProps,\n  ResourceTitle,\n  ResourceTitleLink as StyledLink,\n  ResourceTypeList,\n  StyledContentIconWrapper,\n  LoaderProps,\n} from './resourceComponents';\nimport ContentLoader from '../ContentLoader';\nimport ContentTypeBadge from '../ContentTypeBadge';\nimport { contentTypeMapping } from '../model/ContentType';\nimport { SettingsMenu } from '../MyNdla';\n\nconst ListResourceWrapper = styled.div`\n  flex: 1;\n  display: grid;\n  position: relative;\n  grid-template-columns: auto minmax(50px, 1fr) auto;\n  grid-template-areas:\n    'image  topicAndTitle   tags'\n    'image  description     description';\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  border: 1px solid ${colors.brand.neutral7};\n  border-radius: 2px;\n\n  &:hover {\n    box-shadow: 1px 1px 6px 2px rgba(9, 55, 101, 0.08);\n    transition-duration: 0.2s;\n    ${() => StyledLink} {\n      color: ${colors.brand.primary};\n      text-decoration: underline;\n    }\n  }\n`;\n\ninterface StyledImageProps {\n  imageSize: 'normal' | 'compact';\n}\n\nconst ImageWrapper = styled.div<StyledImageProps>`\n  grid-area: image;\n  width: ${(p) => (p.imageSize === 'normal' ? '136px' : '56px')};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    width: 56px;\n    margin-bottom: 0;\n  }\n  overflow: hidden;\n  border-radius: 2px;\n  display: flex;\n  margin: ${spacing.small};\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 StyledResourceDescription = styled.p`\n  grid-area: description;\n  line-clamp: 2;\n  line-height: 1em;\n  height: 3.1em;\n  margin: 0 ${spacing.small} ${spacing.small} 0;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin: 0 ${spacing.small};\n  }\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 TagsandActionMenu = styled.div`\n  grid-area: tags;\n  z-index: 1;\n  box-sizing: content-box;\n  display: grid;\n  grid-template-columns: 1fr auto auto;\n  align-items: center;\n  align-self: flex-start;\n  justify-items: flex-end;\n  overflow: hidden;\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    min-height: ${spacing.small};\n  }\n`;\n\nconst TopicAndTitleWrapper = styled.div`\n  grid-area: topicAndTitle;\n  display: flex;\n  margin: ${spacing.small} 0;\n  flex-direction: column;\n  overflow: hidden;\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.mobileWide })} {\n    margin-bottom: 0;\n  }\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 TypeAndTitleLoader = ({ 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 Description = ({ 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\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\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 firstContentType = resourceTypes?.[0]?.id ?? '';\n  const Title = ResourceTitle.withComponent(headingLevel);\n\n  return (\n    <ListResourceWrapper id={id}>\n      <ImageWrapper imageSize={imageType}>\n        <ListResourceImage\n          resourceImage={resourceImage}\n          loading={isLoading}\n          type={imageType}\n          contentType={contentTypeMapping[firstContentType] ?? contentTypeMapping['default']}\n        />\n      </ImageWrapper>\n      <TopicAndTitleWrapper>\n        <TypeAndTitleLoader loading={isLoading}>\n          <StyledLink to={link} target={targetBlank ? '_blank' : undefined}>\n            <Title title={title}>{title}</Title>\n          </StyledLink>\n          <ResourceTypeList resourceTypes={resourceTypes} />\n        </TypeAndTitleLoader>\n      </TopicAndTitleWrapper>\n      {showDescription && <Description description={description} loading={isLoading} />}\n      <TagsandActionMenu>\n        {tags && tags.length > 0 && <CompressedTagList tagLinkPrefix={tagLinkPrefix} tags={tags} />}\n        {menuItems && menuItems.length > 0 && <SettingsMenu menuItems={menuItems} />}\n      </TagsandActionMenu>\n    </ListResourceWrapper>\n  );\n};\n\nexport default ListResource;\n"]} */"));
68
68
  var ListResourceImage = function ListResourceImage(_ref) {
69
69
  var resourceImage = _ref.resourceImage,
70
70
  loading = _ref.loading,
@@ -214,9 +214,7 @@ var ListResource = function ListResource(_ref4) {
214
214
  children: [tags && tags.length > 0 && _jsx(CompressedTagList, {
215
215
  tagLinkPrefix: tagLinkPrefix,
216
216
  tags: tags
217
- }), menuItems && menuItems.length > 0 && _jsx(MenuButton, {
218
- align: "end",
219
- size: "small",
217
+ }), menuItems && menuItems.length > 0 && _jsx(SettingsMenu, {
220
218
  menuItems: menuItems
221
219
  })]
222
220
  })]
package/es/index.js CHANGED
@@ -79,7 +79,7 @@ export { default as CopyParagraphButton, CopyParagraphButtonV2 } from './CopyPar
79
79
  export { default as ContentPlaceholder } from './ContentPlaceholder';
80
80
  export { Notion, ConceptNotion } from './Notion';
81
81
  export { BannerCard } from './BannerCard';
82
- export { Folder, FolderInput } from './MyNdla';
82
+ export { Folder, FolderInput, FolderMenu } from './MyNdla';
83
83
  export { ListResource, BlockResource } from './Resource';
84
84
  export { TagSelector } from './TagSelector';
85
85
  export { SnackbarProvider, useSnack, BaseSnack, DefaultSnackbar } from './SnackBar';
@@ -1051,6 +1051,7 @@ var messages = _objectSpread(_objectSpread({
1051
1051
  resources_plural: '{{count}} Resources',
1052
1052
  folders: '{{count}} Folder',
1053
1053
  folders_plural: '{{count}} Folders',
1054
+ settings: 'Settings',
1054
1055
  folder: {
1055
1056
  folder: 'Folder',
1056
1057
  "delete": 'Delete folder',
@@ -1084,7 +1085,7 @@ var messages = _objectSpread(_objectSpread({
1084
1085
  },
1085
1086
  button: {
1086
1087
  share: 'Share folder',
1087
- preview: 'Preview shared folder',
1088
+ preview: 'Preview folder',
1088
1089
  unShare: 'Stop sharing',
1089
1090
  shareLink: 'Copy link'
1090
1091
  }
@@ -1111,6 +1112,7 @@ var messages = _objectSpread(_objectSpread({
1111
1112
  examLockInfo: 'Editing content on Min NDLA is deactivated for pupils during the exam period.',
1112
1113
  help: 'Help',
1113
1114
  more: 'More options',
1115
+ selectView: 'Select view',
1114
1116
  listView: 'List view',
1115
1117
  detailView: 'Detailed listview',
1116
1118
  shortView: 'Card view',
@@ -1051,6 +1051,7 @@ var messages = _objectSpread(_objectSpread({
1051
1051
  resources_plural: '{{count}} ressurser',
1052
1052
  folders: '{{count}} mappe',
1053
1053
  folders_plural: '{{count}} mapper',
1054
+ settings: 'Instillinger',
1054
1055
  folder: {
1055
1056
  folder: 'Mappe',
1056
1057
  "delete": 'Slett mappe',
@@ -1084,7 +1085,7 @@ var messages = _objectSpread(_objectSpread({
1084
1085
  },
1085
1086
  button: {
1086
1087
  share: 'Del mappen',
1087
- preview: 'Forhåndsvis delt mappe',
1088
+ preview: 'Forhåndsvis mappe',
1088
1089
  unShare: 'Avslutt deling',
1089
1090
  shareLink: 'Kopier lenke'
1090
1091
  }
@@ -1111,6 +1112,7 @@ var messages = _objectSpread(_objectSpread({
1111
1112
  examLockInfo: 'Redigering av innhold på Min NDLA er deaktivert for elever i eksamensperioden.',
1112
1113
  help: 'Hjelp',
1113
1114
  more: 'Flere valg',
1115
+ selectView: 'Velg visning',
1114
1116
  listView: 'Listevisning',
1115
1117
  detailView: 'Detaljert listevisning',
1116
1118
  shortView: 'Kort visning',
@@ -1051,6 +1051,7 @@ var messages = _objectSpread(_objectSpread({
1051
1051
  resources_plural: '{{count}} ressursar',
1052
1052
  folders: '{{count}} mappe',
1053
1053
  folders_plural: '{{count}} mapper',
1054
+ settings: 'Instillinger',
1054
1055
  folder: {
1055
1056
  folder: 'Mappe',
1056
1057
  "delete": 'Slett mappe',
@@ -1084,7 +1085,7 @@ var messages = _objectSpread(_objectSpread({
1084
1085
  },
1085
1086
  button: {
1086
1087
  share: 'Del mappa',
1087
- preview: 'Førehandsvis delt mappe',
1088
+ preview: 'Førehandsvis mappe',
1088
1089
  unShare: 'Avslutt deling',
1089
1090
  shareLink: 'Kopier lenke'
1090
1091
  }
@@ -1111,6 +1112,7 @@ var messages = _objectSpread(_objectSpread({
1111
1112
  examLockInfo: 'Redigering av innhald på Min NDLA er deaktivert for elevar i eksamensperioden.',
1112
1113
  help: 'Hjelp',
1113
1114
  more: 'Fleire val',
1115
+ selectView: 'Velg visning',
1114
1116
  listView: 'Listevisning',
1115
1117
  detailView: 'Detaljert listevisning',
1116
1118
  shortView: 'Kortvisning',
@@ -1051,6 +1051,7 @@ var messages = _objectSpread(_objectSpread({
1051
1051
  resources_plural: '{{count}} resurssat',
1052
1052
  folders: '{{count}} máhpat',
1053
1053
  folders_plural: '{{count}} mapper',
1054
+ settings: 'Instillinger',
1054
1055
  folder: {
1055
1056
  folder: 'Máhppa',
1056
1057
  "delete": 'Sihko máhpa',
@@ -1084,7 +1085,7 @@ var messages = _objectSpread(_objectSpread({
1084
1085
  },
1085
1086
  button: {
1086
1087
  share: 'Del mappen',
1087
- preview: 'Forhåndsvis delt mappe',
1088
+ preview: 'Forhåndsvis mappe',
1088
1089
  unShare: 'Avslutt deling',
1089
1090
  shareLink: 'Kopier lenke'
1090
1091
  }
@@ -1111,6 +1112,7 @@ var messages = _objectSpread(_objectSpread({
1111
1112
  examLockInfo: 'Mu NDLA sisdoalu redigeren ii leat doaimmas ohppiide eksámenáigodagas.',
1112
1113
  help: 'Veahkki',
1113
1114
  more: 'Eanet válljejumit',
1115
+ selectView: 'Velg visning',
1114
1116
  listView: 'Oppalašlistu',
1115
1117
  detailView: 'Bienalaš oppalašlistu',
1116
1118
  shortView: 'Oanehis listu',
@@ -1051,6 +1051,7 @@ var messages = _objectSpread(_objectSpread({
1051
1051
  resources_plural: '{{count}} ressurser',
1052
1052
  folders: '{{count}} mappe',
1053
1053
  folders_plural: '{{count}} mapper',
1054
+ settings: 'Instillinger',
1054
1055
  folder: {
1055
1056
  folder: 'Mappe',
1056
1057
  "delete": 'Slett mappe',
@@ -1084,7 +1085,7 @@ var messages = _objectSpread(_objectSpread({
1084
1085
  },
1085
1086
  button: {
1086
1087
  share: 'Del mappen',
1087
- preview: 'Forhåndsvis delt mappe',
1088
+ preview: 'Forhåndsvis mappe',
1088
1089
  unShare: 'Avslutt deling',
1089
1090
  shareLink: 'Kopier lenke'
1090
1091
  }
@@ -1111,6 +1112,7 @@ var messages = _objectSpread(_objectSpread({
1111
1112
  examLockInfo: 'Redigering av innhold på Min NDLA er deaktivert for elever i eksamensperioden.',
1112
1113
  help: 'Hjelp',
1113
1114
  more: 'Flere valg',
1115
+ selectView: 'Velg visning',
1114
1116
  listView: 'Listevisning',
1115
1117
  detailView: 'Detaljert listevisning',
1116
1118
  shortView: 'Kort visning',
@@ -6,7 +6,7 @@
6
6
  *
7
7
  */
8
8
  import { MenuItemProps } from '@ndla/button';
9
- type LayoutType = 'list' | 'listLarger' | 'block';
9
+ export type LayoutType = 'list' | 'listLarger' | 'block';
10
10
  interface Props {
11
11
  id: string;
12
12
  title: string;
@@ -14,9 +14,10 @@ interface Props {
14
14
  subResources?: number;
15
15
  description?: string;
16
16
  link: string;
17
- type: LayoutType;
17
+ type?: LayoutType;
18
+ onViewTypeChange?: (type: LayoutType) => void;
18
19
  menuItems?: MenuItemProps[];
19
20
  isShared?: boolean;
20
21
  }
21
- declare const Folder: ({ id, link, title, subFolders, subResources, type, menuItems, isShared }: Props) => import("@emotion/react/jsx-runtime").JSX.Element;
22
+ declare const Folder: ({ id, link, title, subFolders, subResources, type, menuItems, isShared, onViewTypeChange, }: Props) => import("@emotion/react/jsx-runtime").JSX.Element;
22
23
  export default Folder;