@ndla/ui 16.1.1 → 18.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (126) hide show
  1. package/es/Article/Article.js +3 -3
  2. package/es/Article/ArticleFavoritesButton.js +4 -3
  3. package/es/Breadcrumblist/Breadcrumblist.js +7 -7
  4. package/es/Frontpage/FrontpageAllSubjects.js +8 -8
  5. package/es/Masthead/Masthead.js +2 -3
  6. package/es/Messages/MessageBanner.js +75 -0
  7. package/es/Messages/MessageBox.js +123 -0
  8. package/es/{MessageBox → Messages}/MessageBoxTag.js +1 -1
  9. package/es/{MessageBox → Messages}/index.js +3 -3
  10. package/es/MyNdla/Resource/Folder.js +6 -6
  11. package/es/MyNdla/index.js +1 -2
  12. package/es/Notion/ConceptNotion.js +1 -1
  13. package/es/Notion/index.js +2 -1
  14. package/es/Programme/Programme.js +6 -6
  15. package/es/Programme/ProgrammeSubjects.js +2 -2
  16. package/es/Resource/ListResource.js +6 -6
  17. package/es/TagSelector/SuggestionInput.js +111 -56
  18. package/es/TagSelector/Suggestions.js +19 -15
  19. package/es/TagSelector/TagSelector.js +8 -7
  20. package/es/Topic/Topic.js +22 -24
  21. package/es/TopicMenu/TopicMenu.js +2 -4
  22. package/es/TreeStructure/FolderItem.js +39 -28
  23. package/es/TreeStructure/FolderItems.js +8 -5
  24. package/es/TreeStructure/TreeStructure.js +6 -4
  25. package/es/TreeStructure/TreeStructureWrapper.js +2 -2
  26. package/es/index.js +3 -3
  27. package/es/locale/messages-en.js +9 -2
  28. package/es/locale/messages-nb.js +10 -3
  29. package/es/locale/messages-nn.js +11 -4
  30. package/es/locale/messages-se.js +10 -3
  31. package/es/locale/messages-sma.js +10 -3
  32. package/lib/Article/Article.js +3 -3
  33. package/lib/Article/ArticleFavoritesButton.js +4 -3
  34. package/lib/Breadcrumblist/Breadcrumblist.js +7 -7
  35. package/lib/Breadcrumblist/index.d.ts +1 -0
  36. package/lib/Frontpage/FrontpageAllSubjects.js +9 -9
  37. package/lib/Masthead/Masthead.js +2 -3
  38. package/lib/Messages/MessageBanner.d.ts +16 -0
  39. package/lib/Messages/MessageBanner.js +78 -0
  40. package/lib/{MessageBox → Messages}/MessageBox.d.ts +6 -14
  41. package/lib/Messages/MessageBox.js +128 -0
  42. package/lib/{MessageBox → Messages}/MessageBoxTag.d.ts +0 -0
  43. package/lib/{MessageBox → Messages}/MessageBoxTag.js +1 -1
  44. package/lib/{MessageBox → Messages}/index.d.ts +3 -3
  45. package/lib/Messages/index.js +31 -0
  46. package/lib/MyNdla/Resource/Folder.js +6 -6
  47. package/lib/MyNdla/index.d.ts +1 -2
  48. package/lib/MyNdla/index.js +0 -8
  49. package/lib/Notion/ConceptNotion.d.ts +1 -1
  50. package/lib/Notion/ConceptNotion.js +1 -1
  51. package/lib/Notion/index.d.ts +4 -1
  52. package/lib/Notion/index.js +11 -3
  53. package/lib/Programme/Programme.js +6 -6
  54. package/lib/Programme/ProgrammeSubjects.js +3 -3
  55. package/lib/Resource/ListResource.js +6 -6
  56. package/lib/TagSelector/SuggestionInput.js +111 -57
  57. package/lib/TagSelector/Suggestions.js +26 -23
  58. package/lib/TagSelector/TagSelector.js +8 -7
  59. package/lib/Topic/Topic.js +22 -24
  60. package/lib/TopicMenu/TopicMenu.js +2 -4
  61. package/lib/TreeStructure/FolderItem.d.ts +3 -2
  62. package/lib/TreeStructure/FolderItem.js +38 -28
  63. package/lib/TreeStructure/FolderItems.d.ts +1 -1
  64. package/lib/TreeStructure/FolderItems.js +8 -5
  65. package/lib/TreeStructure/TreeStructure.d.ts +1 -1
  66. package/lib/TreeStructure/TreeStructure.js +6 -4
  67. package/lib/TreeStructure/TreeStructure.types.d.ts +6 -3
  68. package/lib/TreeStructure/TreeStructureWrapper.js +2 -2
  69. package/lib/index.d.ts +8 -4
  70. package/lib/index.js +13 -13
  71. package/lib/locale/messages-en.d.ts +7 -0
  72. package/lib/locale/messages-en.js +9 -2
  73. package/lib/locale/messages-nb.d.ts +7 -0
  74. package/lib/locale/messages-nb.js +10 -3
  75. package/lib/locale/messages-nn.d.ts +7 -0
  76. package/lib/locale/messages-nn.js +11 -4
  77. package/lib/locale/messages-se.d.ts +7 -0
  78. package/lib/locale/messages-se.js +10 -3
  79. package/lib/locale/messages-sma.d.ts +7 -0
  80. package/lib/locale/messages-sma.js +10 -3
  81. package/package.json +13 -13
  82. package/src/Article/Article.tsx +1 -1
  83. package/src/Article/ArticleFavoritesButton.tsx +4 -3
  84. package/src/Breadcrumblist/Breadcrumblist.tsx +1 -1
  85. package/src/Breadcrumblist/{index.tsx → index.ts} +1 -0
  86. package/src/Frontpage/FrontpageAllSubjects.tsx +1 -1
  87. package/src/Masthead/Masthead.tsx +3 -6
  88. package/src/Messages/MessageBanner.tsx +66 -0
  89. package/src/Messages/MessageBox.tsx +156 -0
  90. package/src/{MessageBox → Messages}/MessageBoxTag.tsx +0 -0
  91. package/src/{MessageBox → Messages}/index.ts +3 -3
  92. package/src/MyNdla/Resource/Folder.tsx +1 -1
  93. package/src/MyNdla/index.ts +1 -2
  94. package/src/Notion/ConceptNotion.tsx +2 -1
  95. package/src/Notion/index.ts +4 -1
  96. package/src/Programme/Programme.tsx +1 -1
  97. package/src/Programme/ProgrammeSubjects.tsx +1 -1
  98. package/src/Resource/ListResource.tsx +1 -1
  99. package/src/TagSelector/SuggestionInput.tsx +90 -24
  100. package/src/TagSelector/Suggestions.tsx +14 -0
  101. package/src/TagSelector/TagSelector.tsx +6 -4
  102. package/src/Topic/Topic.tsx +2 -2
  103. package/src/TopicMenu/TopicMenu.jsx +2 -2
  104. package/src/TreeStructure/FolderItem.tsx +43 -19
  105. package/src/TreeStructure/FolderItems.tsx +3 -0
  106. package/src/TreeStructure/TreeStructure.tsx +2 -0
  107. package/src/TreeStructure/TreeStructure.types.ts +7 -3
  108. package/src/TreeStructure/TreeStructureWrapper.tsx +1 -1
  109. package/src/index.ts +17 -4
  110. package/src/locale/messages-en.ts +10 -2
  111. package/src/locale/messages-nb.ts +10 -3
  112. package/src/locale/messages-nn.ts +11 -4
  113. package/src/locale/messages-se.ts +10 -3
  114. package/src/locale/messages-sma.ts +10 -3
  115. package/es/MessageBox/MessageBox.js +0 -220
  116. package/es/MyNdla/Navigation/VerticalNavigation.js +0 -51
  117. package/es/MyNdla/Navigation/index.js +0 -2
  118. package/lib/MessageBox/MessageBox.js +0 -234
  119. package/lib/MessageBox/index.js +0 -35
  120. package/lib/MyNdla/Navigation/VerticalNavigation.d.ts +0 -10
  121. package/lib/MyNdla/Navigation/VerticalNavigation.js +0 -61
  122. package/lib/MyNdla/Navigation/index.d.ts +0 -2
  123. package/lib/MyNdla/Navigation/index.js +0 -15
  124. package/src/MessageBox/MessageBox.tsx +0 -201
  125. package/src/MyNdla/Navigation/VerticalNavigation.tsx +0 -93
  126. package/src/MyNdla/Navigation/index.ts +0 -2
@@ -23,6 +23,10 @@ var titleTemplate = ' - NDLA';
23
23
 
24
24
  var messages = _objectSpread(_objectSpread({
25
25
  treeStructure: {
26
+ folderChildOptions: {
27
+ edit: 'Endre mappenavn',
28
+ "delete": 'Slett'
29
+ },
26
30
  createFolder: 'Lag mappe',
27
31
  newFolder: {
28
32
  placeholder: 'Skriv navn på mappe',
@@ -30,8 +34,9 @@ var messages = _objectSpread(_objectSpread({
30
34
  }
31
35
  },
32
36
  tagSelector: {
33
- placeholder: 'Legg til tag',
34
- removeTag: 'Ta vekk tilknytninga til {{name}}',
37
+ label: 'Legg til tag',
38
+ placeholder: 'Skriv tag navn',
39
+ removeTag: 'Ta vekk {{name}}',
35
40
  hideAllTags: 'Skjul alla tags',
36
41
  showAllTags: 'Vis alla tags'
37
42
  },
@@ -312,7 +317,7 @@ var messages = _objectSpread(_objectSpread({
312
317
  feide: 'Denne ressursen er bare tilgjengelig for lærere som er pålogget med Feide.',
313
318
  resources: 'Dette er ikke et komplett læremiddel produsert av NDLA, men ei ressurssamling som vi håper kan være nyttig for deg.',
314
319
  subjectOutdated: 'Dette faget følger en utgått læreplan.',
315
- subjectBeta: 'Dette faget er i betaversjon.',
320
+ subjectBeta: 'Dette faget er i betaversjon. Vi fyller på med ressurser fortløpende.',
316
321
  newVersion: 'Denne læringsressursen er ikke oppdatert etter gjeldende læreplan. Du finner en oppdatert versjon her: ',
317
322
  frontPageBeta: 'Kommende fag er tilpassa ny læreplan som gjelder fra høsten 2022. Betafag er fag under arbeid. Vi håper likevel at læringsressursene i betafaga kan være nyttige allerede nå.',
318
323
  frontPageExpired: 'Utgåtte fag undervises det ikke i lenger, men det kan fortsatt være mulig å ta eksamen i faga som privatist.',
@@ -945,6 +950,8 @@ var messages = _objectSpread(_objectSpread({
945
950
  newFolderUnder: 'Lag ny mappe under {{folderName}}',
946
951
  myAccount: 'Min konto',
947
952
  favourites: 'Favoritter',
953
+ addToFavourites: 'Legg til i mine favoritter',
954
+ alreadyFavourited: 'Allerede lagt til i mine favoritter',
948
955
  help: 'Hjelp',
949
956
  more: 'Flere valg',
950
957
  listView: 'Listevisning',
@@ -23,6 +23,10 @@ var titleTemplate = ' - NDLA';
23
23
 
24
24
  var messages = _objectSpread(_objectSpread({
25
25
  treeStructure: {
26
+ folderChildOptions: {
27
+ edit: 'Endre mappenavn',
28
+ "delete": 'Slett'
29
+ },
26
30
  createFolder: 'Lag mappe',
27
31
  newFolder: {
28
32
  placeholder: 'Skriv navn på mappe',
@@ -30,8 +34,9 @@ var messages = _objectSpread(_objectSpread({
30
34
  }
31
35
  },
32
36
  tagSelector: {
33
- placeholder: 'Tilknytt tag',
34
- removeTag: 'Ta vekk tilknytning til {{name}}',
37
+ label: 'Legg til tag',
38
+ placeholder: 'Skriv tag navn',
39
+ removeTag: 'Ta vekk {{name}}',
35
40
  hideAllTags: 'Skjul alle tagger',
36
41
  showAllTags: 'Vis alle tagger'
37
42
  },
@@ -312,7 +317,7 @@ var messages = _objectSpread(_objectSpread({
312
317
  feide: 'Denne ressursen er bare tilgjengelig for lærere som er pålogget med Feide.',
313
318
  resources: 'Dette er ikke et komplett læremiddel produsert av NDLA, men ei ressurssamling som vi håper kan være nyttig for deg.',
314
319
  subjectOutdated: 'Dette faget følger en utgått læreplan.',
315
- subjectBeta: 'Dette faget er i betaversjon.',
320
+ subjectBeta: 'Dette faget er i betaversjon. Vi fyller på med ressurser fortløpende.',
316
321
  newVersion: 'Denne læringsressursen er ikke oppdatert etter gjeldende læreplan. Du finner en oppdatert versjon her: ',
317
322
  frontPageBeta: 'Kommende fag er tilpassa ny læreplan som gjelder fra høsten 2022. Betafag er fag under arbeid. Vi håper likevel at læringsressursene i betafaga kan være nyttige allerede nå.',
318
323
  frontPageExpired: 'Utgåtte fag undervises det ikke i lenger, men det kan fortsatt være mulig å ta eksamen i faga som privatist.',
@@ -945,6 +950,8 @@ var messages = _objectSpread(_objectSpread({
945
950
  newFolderUnder: 'Lag ny mappe under {{folderName}}',
946
951
  myAccount: 'Min konto',
947
952
  favourites: 'Favoritter',
953
+ addToFavourites: 'Legg til i mine favoritter',
954
+ alreadyFavourited: 'Allerede lagt til i mine favoritter',
948
955
  help: 'Hjelp',
949
956
  more: 'Flere valg',
950
957
  listView: 'Listevisning',
@@ -37,7 +37,7 @@ var _ArticleNotions = _interopRequireDefault(require("./ArticleNotions"));
37
37
 
38
38
  var _ArticleAccessMessage = _interopRequireDefault(require("./ArticleAccessMessage"));
39
39
 
40
- var _MessageBox = _interopRequireDefault(require("../MessageBox/MessageBox"));
40
+ var _MessageBox = _interopRequireDefault(require("../Messages/MessageBox"));
41
41
 
42
42
  var _core2 = require("@emotion/core");
43
43
 
@@ -136,7 +136,7 @@ var MSGboxWrapper = (0, _styledBase["default"])("div", {
136
136
  } : {
137
137
  name: "1p2cbqg",
138
138
  styles: "margin-bottom:50px;",
139
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Article.tsx"],"names":[],"mappings":"AAiGgC","file":"Article.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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 React, { ComponentType, ReactNode, useEffect, useRef, useState, forwardRef } from 'react';\nimport BEMHelper from 'react-bem-helper';\nimport isString from 'lodash/isString';\nimport parse from 'html-react-parser';\nimport styled from '@emotion/styled';\n\nimport { useIntersectionObserver } from '@ndla/hooks';\nimport { resizeObserver } from '@ndla/util';\nimport { spacing, spacingUnit, mq, breakpoints } from '@ndla/core';\nimport { Article as ArticleType, Locale } from '../types';\nimport ArticleFootNotes from './ArticleFootNotes';\nimport ArticleContent from './ArticleContent';\nimport ArticleByline from './ArticleByline';\nimport ArticleFavoritesButton from './ArticleFavoritesButton';\nimport LayoutItem from '../Layout';\nimport ArticleHeaderWrapper from './ArticleHeaderWrapper';\nimport ArticleNotions, { NotionRelatedContent } from './ArticleNotions';\nimport ArticleAccessMessage from './ArticleAccessMessage';\nimport MessageBox from '../MessageBox/MessageBox';\nimport { ConceptNotionType } from '../Notion/ConceptNotion';\n\nconst classes = new BEMHelper({\n  name: 'article',\n  prefix: 'c-',\n});\n\ntype ArticleWrapperProps = {\n  id: string;\n  modifier?: string;\n  children: ReactNode;\n};\n\nexport const ArticleWrapper = forwardRef<HTMLElement, ArticleWrapperProps>(({ children, modifier, id }, ref) => (\n  <article id={id} {...classes(undefined, modifier)} ref={ref}>\n    {children}\n  </article>\n));\n\ntype ArticleTitleProps = {\n  icon?: ReactNode;\n  label?: string;\n  children: ReactNode;\n};\n\nexport const ArticleTitle = ({ children, icon, label }: ArticleTitleProps) => {\n  const modifiers = [];\n  if (icon) {\n    modifiers.push('icon');\n  }\n\n  let labelView = null;\n\n  if (label) {\n    labelView = <p>{label}</p>;\n  }\n\n  return (\n    <div {...classes('title', modifiers)}>\n      {icon}\n      {labelView}\n      <h1 tabIndex={0}>{children}</h1>\n    </div>\n  );\n};\n\ntype ArticleIntroductionProps = {\n  children: ReactNode;\n  renderMarkdown: (text: string) => string;\n};\n\nexport const ArticleIntroduction = ({\n  children,\n  renderMarkdown = (text) => {\n    return text;\n  },\n}: ArticleIntroductionProps) => {\n  if (isString(children)) {\n    return <p className=\"article_introduction\">{parse(renderMarkdown(children))}</p>;\n  }\n  if (children) {\n    return <p className=\"article_introduction\">{children}</p>;\n  }\n  return null;\n};\n\ntype Messages = {\n  label: string;\n  messageBox?: string;\n};\nconst MSGboxWrapper = styled.div`\n  margin-bottom: 50px;\n`;\n\nconst ArticleFavoritesButtonWrapper = styled.div`\n  display: flex;\n  justify-content: flex-end;\n  transform: translate(${spacing.xsmall}, -${spacing.normal});\n  height: 0;\n  ${mq.range({ from: breakpoints.tablet })} {\n    transform: translate(${spacing.normal}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    transform: translate(${spacing.large}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    transform: translate(${spacingUnit * 5.5}px, -${spacing.medium});\n  }\n`;\n\ntype Props = {\n  article: ArticleType;\n  icon?: ReactNode;\n  licenseBox?: ReactNode;\n  modifier?: string;\n  children?: ReactNode;\n  messages: Messages;\n  locale: Locale;\n  messageBoxLinks?: [];\n  competenceGoals?:\n    | ((inp: { Dialog: ComponentType; dialogProps: { isOpen: boolean; onClose: () => void } }) => ReactNode)\n    | ReactNode\n    | null;\n  competenceGoalTypes?: string[];\n  id: string;\n  renderMarkdown: (text: string) => string;\n  copyPageUrlLink?: string;\n  printUrl?: string;\n  notions?: { list: ConceptNotionType[]; related: NotionRelatedContent[] };\n  accessMessage?: string;\n  isFavorite?: boolean;\n  onToggleAddToFavorites?: (id: string, add: boolean) => void;\n};\n\nconst getArticleContent = (content: any, locale: Locale) => {\n  switch (typeof content) {\n    case 'string':\n      return <ArticleContent content={content} locale={locale} />;\n    case 'function':\n      return content();\n    default:\n      return content;\n  }\n};\n\nexport const Article = ({\n  article,\n  icon,\n  licenseBox,\n  modifier,\n  messages,\n  messageBoxLinks,\n  children,\n  competenceGoals,\n  competenceGoalTypes,\n  copyPageUrlLink,\n  id,\n  locale,\n  notions,\n  printUrl,\n  renderMarkdown,\n  accessMessage,\n  onToggleAddToFavorites,\n  isFavorite,\n}: Props) => {\n  const [articleRef, { entry }] = useIntersectionObserver({\n    root: null,\n    rootMargin: '400px',\n    threshold: 0.1,\n  });\n  const [articlePositionRight, setArticlePositionRight] = useState(0);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n\n  const showExplainNotions = entry && entry.isIntersecting;\n\n  useEffect(() => {\n    if (wrapperRef && wrapperRef.current) {\n      const handler = () => {\n        if (wrapperRef && wrapperRef.current) {\n          const offset =\n            wrapperRef.current.getBoundingClientRect().left + wrapperRef.current.getBoundingClientRect().width;\n          setArticlePositionRight(offset);\n        }\n      };\n      handler();\n\n      return resizeObserver(document.body, handler);\n    }\n  }, [wrapperRef]);\n\n  const {\n    title,\n    introduction,\n    published,\n    content,\n    footNotes,\n    copyright: { license: licenseObj, creators, rightsholders, processors },\n  } = article;\n\n  const authors = creators.length || rightsholders.length ? creators : processors;\n\n  return (\n    <div ref={wrapperRef}>\n      <ArticleWrapper modifier={modifier} id={id} ref={articleRef}>\n        <LayoutItem layout=\"center\">\n          {accessMessage && <ArticleAccessMessage message={accessMessage} />}\n\n          {messages.messageBox && (\n            <MSGboxWrapper>\n              <MessageBox links={messageBoxLinks}>{messages.messageBox}</MessageBox>\n            </MSGboxWrapper>\n          )}\n          <ArticleHeaderWrapper competenceGoals={competenceGoals} competenceGoalTypes={competenceGoalTypes}>\n            {onToggleAddToFavorites && (\n              <ArticleFavoritesButtonWrapper>\n                <ArticleFavoritesButton\n                  articleId={id}\n                  isFavorite={isFavorite}\n                  onToggleAddToFavorites={onToggleAddToFavorites}\n                />\n              </ArticleFavoritesButtonWrapper>\n            )}\n            <ArticleTitle icon={icon} label={messages.label}>\n              {title}\n            </ArticleTitle>\n            <ArticleIntroduction renderMarkdown={renderMarkdown}>{introduction}</ArticleIntroduction>\n          </ArticleHeaderWrapper>\n        </LayoutItem>\n        <LayoutItem layout=\"center\">\n          {notions && showExplainNotions && (\n            <ArticleNotions\n              notions={notions.list}\n              relatedContent={notions.related}\n              buttonOffsetRight={articlePositionRight}\n            />\n          )}\n          {getArticleContent(content, locale)}\n        </LayoutItem>\n\n        <LayoutItem layout=\"center\">\n          {footNotes && footNotes.length > 0 && <ArticleFootNotes footNotes={footNotes} />}\n          <ArticleByline\n            copyPageUrlLink={copyPageUrlLink}\n            authors={authors}\n            suppliers={rightsholders}\n            published={published}\n            license={licenseObj.license}\n            licenseBox={licenseBox}\n            printUrl={printUrl}\n          />\n        </LayoutItem>\n        <LayoutItem layout=\"extend\">{children}</LayoutItem>\n      </ArticleWrapper>\n    </div>\n  );\n};\n\nexport default Article;\n"]} */",
139
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Article.tsx"],"names":[],"mappings":"AAiGgC","file":"Article.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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 React, { ComponentType, ReactNode, useEffect, useRef, useState, forwardRef } from 'react';\nimport BEMHelper from 'react-bem-helper';\nimport isString from 'lodash/isString';\nimport parse from 'html-react-parser';\nimport styled from '@emotion/styled';\n\nimport { useIntersectionObserver } from '@ndla/hooks';\nimport { resizeObserver } from '@ndla/util';\nimport { spacing, spacingUnit, mq, breakpoints } from '@ndla/core';\nimport { Article as ArticleType, Locale } from '../types';\nimport ArticleFootNotes from './ArticleFootNotes';\nimport ArticleContent from './ArticleContent';\nimport ArticleByline from './ArticleByline';\nimport ArticleFavoritesButton from './ArticleFavoritesButton';\nimport LayoutItem from '../Layout';\nimport ArticleHeaderWrapper from './ArticleHeaderWrapper';\nimport ArticleNotions, { NotionRelatedContent } from './ArticleNotions';\nimport ArticleAccessMessage from './ArticleAccessMessage';\nimport MessageBox from '../Messages/MessageBox';\nimport { ConceptNotionType } from '../Notion/ConceptNotion';\n\nconst classes = new BEMHelper({\n  name: 'article',\n  prefix: 'c-',\n});\n\ntype ArticleWrapperProps = {\n  id: string;\n  modifier?: string;\n  children: ReactNode;\n};\n\nexport const ArticleWrapper = forwardRef<HTMLElement, ArticleWrapperProps>(({ children, modifier, id }, ref) => (\n  <article id={id} {...classes(undefined, modifier)} ref={ref}>\n    {children}\n  </article>\n));\n\ntype ArticleTitleProps = {\n  icon?: ReactNode;\n  label?: string;\n  children: ReactNode;\n};\n\nexport const ArticleTitle = ({ children, icon, label }: ArticleTitleProps) => {\n  const modifiers = [];\n  if (icon) {\n    modifiers.push('icon');\n  }\n\n  let labelView = null;\n\n  if (label) {\n    labelView = <p>{label}</p>;\n  }\n\n  return (\n    <div {...classes('title', modifiers)}>\n      {icon}\n      {labelView}\n      <h1 tabIndex={0}>{children}</h1>\n    </div>\n  );\n};\n\ntype ArticleIntroductionProps = {\n  children: ReactNode;\n  renderMarkdown: (text: string) => string;\n};\n\nexport const ArticleIntroduction = ({\n  children,\n  renderMarkdown = (text) => {\n    return text;\n  },\n}: ArticleIntroductionProps) => {\n  if (isString(children)) {\n    return <p className=\"article_introduction\">{parse(renderMarkdown(children))}</p>;\n  }\n  if (children) {\n    return <p className=\"article_introduction\">{children}</p>;\n  }\n  return null;\n};\n\ntype Messages = {\n  label: string;\n  messageBox?: string;\n};\nconst MSGboxWrapper = styled.div`\n  margin-bottom: 50px;\n`;\n\nconst ArticleFavoritesButtonWrapper = styled.div`\n  display: flex;\n  justify-content: flex-end;\n  transform: translate(${spacing.xsmall}, -${spacing.normal});\n  height: 0;\n  ${mq.range({ from: breakpoints.tablet })} {\n    transform: translate(${spacing.normal}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    transform: translate(${spacing.large}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    transform: translate(${spacingUnit * 5.5}px, -${spacing.medium});\n  }\n`;\n\ntype Props = {\n  article: ArticleType;\n  icon?: ReactNode;\n  licenseBox?: ReactNode;\n  modifier?: string;\n  children?: ReactNode;\n  messages: Messages;\n  locale: Locale;\n  messageBoxLinks?: [];\n  competenceGoals?:\n    | ((inp: { Dialog: ComponentType; dialogProps: { isOpen: boolean; onClose: () => void } }) => ReactNode)\n    | ReactNode\n    | null;\n  competenceGoalTypes?: string[];\n  id: string;\n  renderMarkdown: (text: string) => string;\n  copyPageUrlLink?: string;\n  printUrl?: string;\n  notions?: { list: ConceptNotionType[]; related: NotionRelatedContent[] };\n  accessMessage?: string;\n  isFavorite?: boolean;\n  onToggleAddToFavorites?: (id: string, add: boolean) => void;\n};\n\nconst getArticleContent = (content: any, locale: Locale) => {\n  switch (typeof content) {\n    case 'string':\n      return <ArticleContent content={content} locale={locale} />;\n    case 'function':\n      return content();\n    default:\n      return content;\n  }\n};\n\nexport const Article = ({\n  article,\n  icon,\n  licenseBox,\n  modifier,\n  messages,\n  messageBoxLinks,\n  children,\n  competenceGoals,\n  competenceGoalTypes,\n  copyPageUrlLink,\n  id,\n  locale,\n  notions,\n  printUrl,\n  renderMarkdown,\n  accessMessage,\n  onToggleAddToFavorites,\n  isFavorite,\n}: Props) => {\n  const [articleRef, { entry }] = useIntersectionObserver({\n    root: null,\n    rootMargin: '400px',\n    threshold: 0.1,\n  });\n  const [articlePositionRight, setArticlePositionRight] = useState(0);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n\n  const showExplainNotions = entry && entry.isIntersecting;\n\n  useEffect(() => {\n    if (wrapperRef && wrapperRef.current) {\n      const handler = () => {\n        if (wrapperRef && wrapperRef.current) {\n          const offset =\n            wrapperRef.current.getBoundingClientRect().left + wrapperRef.current.getBoundingClientRect().width;\n          setArticlePositionRight(offset);\n        }\n      };\n      handler();\n\n      return resizeObserver(document.body, handler);\n    }\n  }, [wrapperRef]);\n\n  const {\n    title,\n    introduction,\n    published,\n    content,\n    footNotes,\n    copyright: { license: licenseObj, creators, rightsholders, processors },\n  } = article;\n\n  const authors = creators.length || rightsholders.length ? creators : processors;\n\n  return (\n    <div ref={wrapperRef}>\n      <ArticleWrapper modifier={modifier} id={id} ref={articleRef}>\n        <LayoutItem layout=\"center\">\n          {accessMessage && <ArticleAccessMessage message={accessMessage} />}\n\n          {messages.messageBox && (\n            <MSGboxWrapper>\n              <MessageBox links={messageBoxLinks}>{messages.messageBox}</MessageBox>\n            </MSGboxWrapper>\n          )}\n          <ArticleHeaderWrapper competenceGoals={competenceGoals} competenceGoalTypes={competenceGoalTypes}>\n            {onToggleAddToFavorites && (\n              <ArticleFavoritesButtonWrapper>\n                <ArticleFavoritesButton\n                  articleId={id}\n                  isFavorite={isFavorite}\n                  onToggleAddToFavorites={onToggleAddToFavorites}\n                />\n              </ArticleFavoritesButtonWrapper>\n            )}\n            <ArticleTitle icon={icon} label={messages.label}>\n              {title}\n            </ArticleTitle>\n            <ArticleIntroduction renderMarkdown={renderMarkdown}>{introduction}</ArticleIntroduction>\n          </ArticleHeaderWrapper>\n        </LayoutItem>\n        <LayoutItem layout=\"center\">\n          {notions && showExplainNotions && (\n            <ArticleNotions\n              notions={notions.list}\n              relatedContent={notions.related}\n              buttonOffsetRight={articlePositionRight}\n            />\n          )}\n          {getArticleContent(content, locale)}\n        </LayoutItem>\n\n        <LayoutItem layout=\"center\">\n          {footNotes && footNotes.length > 0 && <ArticleFootNotes footNotes={footNotes} />}\n          <ArticleByline\n            copyPageUrlLink={copyPageUrlLink}\n            authors={authors}\n            suppliers={rightsholders}\n            published={published}\n            license={licenseObj.license}\n            licenseBox={licenseBox}\n            printUrl={printUrl}\n          />\n        </LayoutItem>\n        <LayoutItem layout=\"extend\">{children}</LayoutItem>\n      </ArticleWrapper>\n    </div>\n  );\n};\n\nexport default Article;\n"]} */",
140
140
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
141
141
  });
142
142
  var ArticleFavoritesButtonWrapper = (0, _styledBase["default"])("div", {
@@ -148,7 +148,7 @@ var ArticleFavoritesButtonWrapper = (0, _styledBase["default"])("div", {
148
148
  from: _core.breakpoints.tabletWide
149
149
  }), "{transform:translate(", _core.spacing.large, ",-", _core.spacing.medium, ");}", _core.mq.range({
150
150
  from: _core.breakpoints.desktop
151
- }), "{transform:translate(", _core.spacingUnit * 5.5, "px,-", _core.spacing.medium, ");}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Article.tsx"],"names":[],"mappings":"AAqGgD","file":"Article.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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 React, { ComponentType, ReactNode, useEffect, useRef, useState, forwardRef } from 'react';\nimport BEMHelper from 'react-bem-helper';\nimport isString from 'lodash/isString';\nimport parse from 'html-react-parser';\nimport styled from '@emotion/styled';\n\nimport { useIntersectionObserver } from '@ndla/hooks';\nimport { resizeObserver } from '@ndla/util';\nimport { spacing, spacingUnit, mq, breakpoints } from '@ndla/core';\nimport { Article as ArticleType, Locale } from '../types';\nimport ArticleFootNotes from './ArticleFootNotes';\nimport ArticleContent from './ArticleContent';\nimport ArticleByline from './ArticleByline';\nimport ArticleFavoritesButton from './ArticleFavoritesButton';\nimport LayoutItem from '../Layout';\nimport ArticleHeaderWrapper from './ArticleHeaderWrapper';\nimport ArticleNotions, { NotionRelatedContent } from './ArticleNotions';\nimport ArticleAccessMessage from './ArticleAccessMessage';\nimport MessageBox from '../MessageBox/MessageBox';\nimport { ConceptNotionType } from '../Notion/ConceptNotion';\n\nconst classes = new BEMHelper({\n  name: 'article',\n  prefix: 'c-',\n});\n\ntype ArticleWrapperProps = {\n  id: string;\n  modifier?: string;\n  children: ReactNode;\n};\n\nexport const ArticleWrapper = forwardRef<HTMLElement, ArticleWrapperProps>(({ children, modifier, id }, ref) => (\n  <article id={id} {...classes(undefined, modifier)} ref={ref}>\n    {children}\n  </article>\n));\n\ntype ArticleTitleProps = {\n  icon?: ReactNode;\n  label?: string;\n  children: ReactNode;\n};\n\nexport const ArticleTitle = ({ children, icon, label }: ArticleTitleProps) => {\n  const modifiers = [];\n  if (icon) {\n    modifiers.push('icon');\n  }\n\n  let labelView = null;\n\n  if (label) {\n    labelView = <p>{label}</p>;\n  }\n\n  return (\n    <div {...classes('title', modifiers)}>\n      {icon}\n      {labelView}\n      <h1 tabIndex={0}>{children}</h1>\n    </div>\n  );\n};\n\ntype ArticleIntroductionProps = {\n  children: ReactNode;\n  renderMarkdown: (text: string) => string;\n};\n\nexport const ArticleIntroduction = ({\n  children,\n  renderMarkdown = (text) => {\n    return text;\n  },\n}: ArticleIntroductionProps) => {\n  if (isString(children)) {\n    return <p className=\"article_introduction\">{parse(renderMarkdown(children))}</p>;\n  }\n  if (children) {\n    return <p className=\"article_introduction\">{children}</p>;\n  }\n  return null;\n};\n\ntype Messages = {\n  label: string;\n  messageBox?: string;\n};\nconst MSGboxWrapper = styled.div`\n  margin-bottom: 50px;\n`;\n\nconst ArticleFavoritesButtonWrapper = styled.div`\n  display: flex;\n  justify-content: flex-end;\n  transform: translate(${spacing.xsmall}, -${spacing.normal});\n  height: 0;\n  ${mq.range({ from: breakpoints.tablet })} {\n    transform: translate(${spacing.normal}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    transform: translate(${spacing.large}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    transform: translate(${spacingUnit * 5.5}px, -${spacing.medium});\n  }\n`;\n\ntype Props = {\n  article: ArticleType;\n  icon?: ReactNode;\n  licenseBox?: ReactNode;\n  modifier?: string;\n  children?: ReactNode;\n  messages: Messages;\n  locale: Locale;\n  messageBoxLinks?: [];\n  competenceGoals?:\n    | ((inp: { Dialog: ComponentType; dialogProps: { isOpen: boolean; onClose: () => void } }) => ReactNode)\n    | ReactNode\n    | null;\n  competenceGoalTypes?: string[];\n  id: string;\n  renderMarkdown: (text: string) => string;\n  copyPageUrlLink?: string;\n  printUrl?: string;\n  notions?: { list: ConceptNotionType[]; related: NotionRelatedContent[] };\n  accessMessage?: string;\n  isFavorite?: boolean;\n  onToggleAddToFavorites?: (id: string, add: boolean) => void;\n};\n\nconst getArticleContent = (content: any, locale: Locale) => {\n  switch (typeof content) {\n    case 'string':\n      return <ArticleContent content={content} locale={locale} />;\n    case 'function':\n      return content();\n    default:\n      return content;\n  }\n};\n\nexport const Article = ({\n  article,\n  icon,\n  licenseBox,\n  modifier,\n  messages,\n  messageBoxLinks,\n  children,\n  competenceGoals,\n  competenceGoalTypes,\n  copyPageUrlLink,\n  id,\n  locale,\n  notions,\n  printUrl,\n  renderMarkdown,\n  accessMessage,\n  onToggleAddToFavorites,\n  isFavorite,\n}: Props) => {\n  const [articleRef, { entry }] = useIntersectionObserver({\n    root: null,\n    rootMargin: '400px',\n    threshold: 0.1,\n  });\n  const [articlePositionRight, setArticlePositionRight] = useState(0);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n\n  const showExplainNotions = entry && entry.isIntersecting;\n\n  useEffect(() => {\n    if (wrapperRef && wrapperRef.current) {\n      const handler = () => {\n        if (wrapperRef && wrapperRef.current) {\n          const offset =\n            wrapperRef.current.getBoundingClientRect().left + wrapperRef.current.getBoundingClientRect().width;\n          setArticlePositionRight(offset);\n        }\n      };\n      handler();\n\n      return resizeObserver(document.body, handler);\n    }\n  }, [wrapperRef]);\n\n  const {\n    title,\n    introduction,\n    published,\n    content,\n    footNotes,\n    copyright: { license: licenseObj, creators, rightsholders, processors },\n  } = article;\n\n  const authors = creators.length || rightsholders.length ? creators : processors;\n\n  return (\n    <div ref={wrapperRef}>\n      <ArticleWrapper modifier={modifier} id={id} ref={articleRef}>\n        <LayoutItem layout=\"center\">\n          {accessMessage && <ArticleAccessMessage message={accessMessage} />}\n\n          {messages.messageBox && (\n            <MSGboxWrapper>\n              <MessageBox links={messageBoxLinks}>{messages.messageBox}</MessageBox>\n            </MSGboxWrapper>\n          )}\n          <ArticleHeaderWrapper competenceGoals={competenceGoals} competenceGoalTypes={competenceGoalTypes}>\n            {onToggleAddToFavorites && (\n              <ArticleFavoritesButtonWrapper>\n                <ArticleFavoritesButton\n                  articleId={id}\n                  isFavorite={isFavorite}\n                  onToggleAddToFavorites={onToggleAddToFavorites}\n                />\n              </ArticleFavoritesButtonWrapper>\n            )}\n            <ArticleTitle icon={icon} label={messages.label}>\n              {title}\n            </ArticleTitle>\n            <ArticleIntroduction renderMarkdown={renderMarkdown}>{introduction}</ArticleIntroduction>\n          </ArticleHeaderWrapper>\n        </LayoutItem>\n        <LayoutItem layout=\"center\">\n          {notions && showExplainNotions && (\n            <ArticleNotions\n              notions={notions.list}\n              relatedContent={notions.related}\n              buttonOffsetRight={articlePositionRight}\n            />\n          )}\n          {getArticleContent(content, locale)}\n        </LayoutItem>\n\n        <LayoutItem layout=\"center\">\n          {footNotes && footNotes.length > 0 && <ArticleFootNotes footNotes={footNotes} />}\n          <ArticleByline\n            copyPageUrlLink={copyPageUrlLink}\n            authors={authors}\n            suppliers={rightsholders}\n            published={published}\n            license={licenseObj.license}\n            licenseBox={licenseBox}\n            printUrl={printUrl}\n          />\n        </LayoutItem>\n        <LayoutItem layout=\"extend\">{children}</LayoutItem>\n      </ArticleWrapper>\n    </div>\n  );\n};\n\nexport default Article;\n"]} */"));
151
+ }), "{transform:translate(", _core.spacingUnit * 5.5, "px,-", _core.spacing.medium, ");}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Article.tsx"],"names":[],"mappings":"AAqGgD","file":"Article.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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 React, { ComponentType, ReactNode, useEffect, useRef, useState, forwardRef } from 'react';\nimport BEMHelper from 'react-bem-helper';\nimport isString from 'lodash/isString';\nimport parse from 'html-react-parser';\nimport styled from '@emotion/styled';\n\nimport { useIntersectionObserver } from '@ndla/hooks';\nimport { resizeObserver } from '@ndla/util';\nimport { spacing, spacingUnit, mq, breakpoints } from '@ndla/core';\nimport { Article as ArticleType, Locale } from '../types';\nimport ArticleFootNotes from './ArticleFootNotes';\nimport ArticleContent from './ArticleContent';\nimport ArticleByline from './ArticleByline';\nimport ArticleFavoritesButton from './ArticleFavoritesButton';\nimport LayoutItem from '../Layout';\nimport ArticleHeaderWrapper from './ArticleHeaderWrapper';\nimport ArticleNotions, { NotionRelatedContent } from './ArticleNotions';\nimport ArticleAccessMessage from './ArticleAccessMessage';\nimport MessageBox from '../Messages/MessageBox';\nimport { ConceptNotionType } from '../Notion/ConceptNotion';\n\nconst classes = new BEMHelper({\n  name: 'article',\n  prefix: 'c-',\n});\n\ntype ArticleWrapperProps = {\n  id: string;\n  modifier?: string;\n  children: ReactNode;\n};\n\nexport const ArticleWrapper = forwardRef<HTMLElement, ArticleWrapperProps>(({ children, modifier, id }, ref) => (\n  <article id={id} {...classes(undefined, modifier)} ref={ref}>\n    {children}\n  </article>\n));\n\ntype ArticleTitleProps = {\n  icon?: ReactNode;\n  label?: string;\n  children: ReactNode;\n};\n\nexport const ArticleTitle = ({ children, icon, label }: ArticleTitleProps) => {\n  const modifiers = [];\n  if (icon) {\n    modifiers.push('icon');\n  }\n\n  let labelView = null;\n\n  if (label) {\n    labelView = <p>{label}</p>;\n  }\n\n  return (\n    <div {...classes('title', modifiers)}>\n      {icon}\n      {labelView}\n      <h1 tabIndex={0}>{children}</h1>\n    </div>\n  );\n};\n\ntype ArticleIntroductionProps = {\n  children: ReactNode;\n  renderMarkdown: (text: string) => string;\n};\n\nexport const ArticleIntroduction = ({\n  children,\n  renderMarkdown = (text) => {\n    return text;\n  },\n}: ArticleIntroductionProps) => {\n  if (isString(children)) {\n    return <p className=\"article_introduction\">{parse(renderMarkdown(children))}</p>;\n  }\n  if (children) {\n    return <p className=\"article_introduction\">{children}</p>;\n  }\n  return null;\n};\n\ntype Messages = {\n  label: string;\n  messageBox?: string;\n};\nconst MSGboxWrapper = styled.div`\n  margin-bottom: 50px;\n`;\n\nconst ArticleFavoritesButtonWrapper = styled.div`\n  display: flex;\n  justify-content: flex-end;\n  transform: translate(${spacing.xsmall}, -${spacing.normal});\n  height: 0;\n  ${mq.range({ from: breakpoints.tablet })} {\n    transform: translate(${spacing.normal}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    transform: translate(${spacing.large}, -${spacing.medium});\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    transform: translate(${spacingUnit * 5.5}px, -${spacing.medium});\n  }\n`;\n\ntype Props = {\n  article: ArticleType;\n  icon?: ReactNode;\n  licenseBox?: ReactNode;\n  modifier?: string;\n  children?: ReactNode;\n  messages: Messages;\n  locale: Locale;\n  messageBoxLinks?: [];\n  competenceGoals?:\n    | ((inp: { Dialog: ComponentType; dialogProps: { isOpen: boolean; onClose: () => void } }) => ReactNode)\n    | ReactNode\n    | null;\n  competenceGoalTypes?: string[];\n  id: string;\n  renderMarkdown: (text: string) => string;\n  copyPageUrlLink?: string;\n  printUrl?: string;\n  notions?: { list: ConceptNotionType[]; related: NotionRelatedContent[] };\n  accessMessage?: string;\n  isFavorite?: boolean;\n  onToggleAddToFavorites?: (id: string, add: boolean) => void;\n};\n\nconst getArticleContent = (content: any, locale: Locale) => {\n  switch (typeof content) {\n    case 'string':\n      return <ArticleContent content={content} locale={locale} />;\n    case 'function':\n      return content();\n    default:\n      return content;\n  }\n};\n\nexport const Article = ({\n  article,\n  icon,\n  licenseBox,\n  modifier,\n  messages,\n  messageBoxLinks,\n  children,\n  competenceGoals,\n  competenceGoalTypes,\n  copyPageUrlLink,\n  id,\n  locale,\n  notions,\n  printUrl,\n  renderMarkdown,\n  accessMessage,\n  onToggleAddToFavorites,\n  isFavorite,\n}: Props) => {\n  const [articleRef, { entry }] = useIntersectionObserver({\n    root: null,\n    rootMargin: '400px',\n    threshold: 0.1,\n  });\n  const [articlePositionRight, setArticlePositionRight] = useState(0);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n\n  const showExplainNotions = entry && entry.isIntersecting;\n\n  useEffect(() => {\n    if (wrapperRef && wrapperRef.current) {\n      const handler = () => {\n        if (wrapperRef && wrapperRef.current) {\n          const offset =\n            wrapperRef.current.getBoundingClientRect().left + wrapperRef.current.getBoundingClientRect().width;\n          setArticlePositionRight(offset);\n        }\n      };\n      handler();\n\n      return resizeObserver(document.body, handler);\n    }\n  }, [wrapperRef]);\n\n  const {\n    title,\n    introduction,\n    published,\n    content,\n    footNotes,\n    copyright: { license: licenseObj, creators, rightsholders, processors },\n  } = article;\n\n  const authors = creators.length || rightsholders.length ? creators : processors;\n\n  return (\n    <div ref={wrapperRef}>\n      <ArticleWrapper modifier={modifier} id={id} ref={articleRef}>\n        <LayoutItem layout=\"center\">\n          {accessMessage && <ArticleAccessMessage message={accessMessage} />}\n\n          {messages.messageBox && (\n            <MSGboxWrapper>\n              <MessageBox links={messageBoxLinks}>{messages.messageBox}</MessageBox>\n            </MSGboxWrapper>\n          )}\n          <ArticleHeaderWrapper competenceGoals={competenceGoals} competenceGoalTypes={competenceGoalTypes}>\n            {onToggleAddToFavorites && (\n              <ArticleFavoritesButtonWrapper>\n                <ArticleFavoritesButton\n                  articleId={id}\n                  isFavorite={isFavorite}\n                  onToggleAddToFavorites={onToggleAddToFavorites}\n                />\n              </ArticleFavoritesButtonWrapper>\n            )}\n            <ArticleTitle icon={icon} label={messages.label}>\n              {title}\n            </ArticleTitle>\n            <ArticleIntroduction renderMarkdown={renderMarkdown}>{introduction}</ArticleIntroduction>\n          </ArticleHeaderWrapper>\n        </LayoutItem>\n        <LayoutItem layout=\"center\">\n          {notions && showExplainNotions && (\n            <ArticleNotions\n              notions={notions.list}\n              relatedContent={notions.related}\n              buttonOffsetRight={articlePositionRight}\n            />\n          )}\n          {getArticleContent(content, locale)}\n        </LayoutItem>\n\n        <LayoutItem layout=\"center\">\n          {footNotes && footNotes.length > 0 && <ArticleFootNotes footNotes={footNotes} />}\n          <ArticleByline\n            copyPageUrlLink={copyPageUrlLink}\n            authors={authors}\n            suppliers={rightsholders}\n            published={published}\n            license={licenseObj.license}\n            licenseBox={licenseBox}\n            printUrl={printUrl}\n          />\n        </LayoutItem>\n        <LayoutItem layout=\"extend\">{children}</LayoutItem>\n      </ArticleWrapper>\n    </div>\n  );\n};\n\nexport default Article;\n"]} */"));
152
152
 
153
153
  var getArticleContent = function getArticleContent(content, locale) {
154
154
  switch (_typeof(content)) {
@@ -37,14 +37,15 @@ var ArticleFavoritesButton = function ArticleFavoritesButton(_ref) {
37
37
  var removeFromFavoritesLabel = t('myNdla.resource.addToMyNdla');
38
38
  var addToFavoritesLabel = t('myNdla.resource.addedToMyNdla');
39
39
  return (0, _core.jsx)(_tooltip["default"], {
40
- tooltip: isFavorite ? removeFromFavoritesLabel : addToFavoritesLabel
40
+ tooltip: isFavorite ? addToFavoritesLabel : removeFromFavoritesLabel
41
41
  }, (0, _core.jsx)(_button.IconButtonDualStates, {
42
- ariaLabelActive: addToFavoritesLabel,
43
- ariaLabelInActive: removeFromFavoritesLabel,
42
+ ariaLabelActive: t('myNdla.addToFavourites'),
43
+ ariaLabelInActive: t('myNdla.alreadyFavourited'),
44
44
  activeIcon: (0, _core.jsx)(_action.Heart, null),
45
45
  inactiveIcon: (0, _core.jsx)(_action.HeartOutline, null),
46
46
  active: isFavorite,
47
47
  size: "small",
48
+ ghostPill: true,
48
49
  onClick: function onClick() {
49
50
  return onToggleAddToFavorites(articleId, !isFavorite);
50
51
  }
@@ -19,7 +19,7 @@ var _safelink = _interopRequireDefault(require("@ndla/safelink"));
19
19
 
20
20
  var _reactI18next = require("react-i18next");
21
21
 
22
- var _MessageBoxTag = _interopRequireDefault(require("../MessageBox/MessageBoxTag"));
22
+ var _MessageBoxTag = _interopRequireDefault(require("../Messages/MessageBoxTag"));
23
23
 
24
24
  var _Masthead = require("../Masthead");
25
25
 
@@ -68,13 +68,13 @@ var Wrapper = (0, _styledBase["default"])("div", {
68
68
  from: '1440px'
69
69
  }), "{margin-left:52px;left:calc((100vw - 1480px) / 2);", function (props) {
70
70
  return props.leftAlign && "\n left: 0;\n ";
71
- }, "}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAmCwC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../MessageBox/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */"));
71
+ }, "}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAmCwC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../Messages/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */"));
72
72
  var Heading = (0, _styledBase["default"])("div", {
73
73
  target: "eblhk4x1",
74
74
  label: "Heading"
75
75
  })("font-weight:bold;font-size:12px;line-height:15px;text-transform:uppercase;padding:0 0 10px 10px;", function (props) {
76
76
  return props.invertedStyle && "\n color: #fff;\n ";
77
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAoFyC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../MessageBox/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */");
77
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAoFyC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../Messages/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */");
78
78
  var List = (0, _styledBase["default"])("ul", {
79
79
  target: "eblhk4x2",
80
80
  label: "List"
@@ -84,7 +84,7 @@ var List = (0, _styledBase["default"])("ul", {
84
84
  } : {
85
85
  name: "ibs9pp",
86
86
  styles: "margin:0 0 20px;padding:0;list-style:none;width:100%;",
87
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAiGsB","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../MessageBox/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */",
87
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAiGsB","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../Messages/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */",
88
88
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
89
89
  });
90
90
  var ListItem = (0, _styledBase["default"])("li", {
@@ -94,7 +94,7 @@ var ListItem = (0, _styledBase["default"])("li", {
94
94
  return props.invertedStyle && "\n color: white;\n ";
95
95
  }, " &:hover{text-decoration:underline;color:", _core.colors.brand.primary, ";", function (props) {
96
96
  return props.invertedStyle && "\n color: #fff;\n ";
97
- }, "}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAwGyC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../MessageBox/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */"));
97
+ }, "}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAwGyC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../Messages/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */"));
98
98
  var IconWrapper = (0, _styledBase["default"])("span", {
99
99
  target: "eblhk4x4",
100
100
  label: "IconWrapper"
@@ -102,13 +102,13 @@ var IconWrapper = (0, _styledBase["default"])("span", {
102
102
  return props.isCurrent && "\n color: ".concat(_core.colors.brand.primary, ";\n ");
103
103
  }, " ", function (props) {
104
104
  return props.invertedStyle && "\n color: #fff;\n ";
105
- }, " .crumbicon{width:24px;height:24px;margin-top:-4px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AA8I0C","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../MessageBox/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */"));
105
+ }, " .crumbicon{width:24px;height:24px;margin-top:-4px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AA8I0C","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../Messages/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */"));
106
106
  var Dot = (0, _styledBase["default"])("span", {
107
107
  target: "eblhk4x5",
108
108
  label: "Dot"
109
109
  })("height:10px;width:10px;background-color:#20588f;border-radius:50%;display:inline-block;margin-left:-15px;margin-right:5px;", function (props) {
110
110
  return props.invertedStyle && "\n background-color: #fff;\n ";
111
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAqKsC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../MessageBox/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */");
111
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Breadcrumblist.tsx"],"names":[],"mappings":"AAqKsC","file":"Breadcrumblist.tsx","sourcesContent":["/**\n * Copyright (c) 2020-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 React, { ReactNode, useEffect, useState, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { mq, breakpoints, colors } from '@ndla/core';\nimport {\n  School as SchoolIcon,\n  MenuBook as MenuBookIcon,\n  Bookmark as BookmarkIcon,\n  Class as ClassIcon,\n  Home as HomeIcon,\n} from '@ndla/icons/action';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport MessageBoxTag from '../Messages/MessageBoxTag';\nimport { useMastheadHeight } from '../Masthead';\n\ntype WrapperProps = {\n  startOffset?: number;\n  isVisible?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  mastheadHeight?: number;\n};\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div<WrapperProps>`\n  display: flex;\n  align-items: flex-start;\n  justify-content: flex-start;\n  flex-direction: column;\n  margin: 32px 0 16px;\n  width: auto;\n  z-index: 1;\n  ${(props) => props.hideOnNarrow && `display:none;`}\n  ${mq.range({ from: breakpoints.wide })} {\n    display: flex;\n    margin: 32px 0;\n    width: 240px;\n    position: fixed;\n    left: 22px;\n    top: ${(props) => props.mastheadHeight || 85}px;\n    ${(props) =>\n      props.startOffset &&\n      `\n        top: calc(${props.startOffset}px + ${props.mastheadHeight || 85}px); \n    `}\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${(props) =>\n      !props.isVisible &&\n      `\n    opacity: 0;\n    transition: opacity 125ms ease-in-out;\n  `}\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    margin: 32px 0;\n    left: 52px;\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n  ${mq.range({ from: '1440px' })} {\n    margin-left: 52px;\n    left: calc((100vw - 1480px) / 2);\n    ${(props) =>\n      props.leftAlign &&\n      `\n        left: 0;\n    `}\n  }\n`;\nconst Heading = styled.div<InvertItProps>`\n  font-weight: bold;\n  font-size: 12px;\n  line-height: 15px;\n  text-transform: uppercase;\n  padding: 0 0 10px 10px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n`;\n\nconst List = styled.ul`\n  margin: 0 0 20px;\n  padding: 0;\n  list-style: none;\n  width: 100%;\n`;\n\nconst ListItem = styled.li<InvertItProps>`\n  font-size: 16px;\n  position: relative;\n  display: flex;\n  flex-direction: row;\n  align-items: center;\n  margin-bottom: 8px;\n  a {\n    color: ${colors.brand.primary};\n    text-decoration: none;\n    box-shadow: none;\n    display: inline-block;\n    width: 99%;\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\n    ${(props) =>\n      props.invertedStyle &&\n      `\n        color: white;\n    `}\n    &:hover {\n      text-decoration: underline;\n      color: ${colors.brand.primary};\n      ${(props) =>\n        props.invertedStyle &&\n        `\n          color: #fff;\n      `}\n    }\n  }\n`;\n\ntype IconProps = {\n  isCurrent: boolean;\n  invertedStyle?: boolean;\n};\n\nconst IconWrapper = styled.span<IconProps>`\n  margin: 0 8px;\n  color: ${colors.brand.tertiary};\n  display: inline-block;\n  min-width: 24px;\n  text-align: center;\n  ${(props) =>\n    props.isCurrent &&\n    `\n    color: ${colors.brand.primary};\n  `}\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      color: #fff;\n  `}\n  .crumbicon {\n    width: 24px;\n    height: 24px;\n    margin-top: -4px;\n  }\n`;\n\nconst Dot = styled.span<InvertItProps>`\n  height: 10px;\n  width: 10px;\n  background-color: #20588f;\n  border-radius: 50%;\n  display: inline-block;\n  margin-left: -15px;\n  margin-right: 5px;\n  ${(props) =>\n    props.invertedStyle &&\n    `\n      background-color: #fff;\n  `}\n`;\n\nconst TypeIcon = (type: string) => {\n  switch (type) {\n    case 'Subjecttype':\n      return <SchoolIcon className=\"crumbicon\" />;\n    case 'Subject':\n      return <MenuBookIcon className=\"crumbicon\" />;\n    case 'Topic':\n      return <BookmarkIcon className=\"crumbicon\" />;\n    case 'Subtopic':\n    case 'SubSubtopic':\n      return <ClassIcon className=\"crumbicon\" />;\n    case 'Home':\n      return <HomeIcon className=\"crumbicon\" />;\n    default:\n      return null;\n  }\n};\n\nexport type BreadcrumbItemProps = {\n  id: string | number;\n  label: string;\n  url: string;\n  typename?: 'Subjecttype' | 'Subject' | 'Topic' | 'Subtopic' | 'SubSubtopic' | 'Home';\n  isCurrent?: boolean | false;\n  icon?: ReactNode;\n};\n\ntype BreadCrumbProps = {\n  children?: ReactNode;\n  items: BreadcrumbItemProps[];\n  startOffset?: number;\n  isVisible?: boolean;\n  invertedStyle?: boolean;\n  leftAlign?: boolean;\n  hideOnNarrow?: boolean;\n  onNav?: (e: MouseEvent<HTMLElement>, item: BreadcrumbItemProps) => void;\n  messageBoxTagMessage?: string;\n};\n\nconst Breadcrumblist = ({\n  children,\n  items,\n  startOffset = 0,\n  isVisible = true,\n  invertedStyle,\n  leftAlign,\n  hideOnNarrow,\n  onNav,\n  messageBoxTagMessage,\n}: BreadCrumbProps) => {\n  const { t } = useTranslation();\n  const [wrapperOffset, setWrapperOffset] = useState(startOffset);\n  const [useScrollEvent, setUseScrollEvent] = useState(false);\n\n  const { height: mastheadHeight } = useMastheadHeight();\n\n  useEffect(() => {\n    const handleScroll = () => {\n      let position = 0;\n      if (window.pageYOffset < startOffset) {\n        position = startOffset;\n      }\n      setWrapperOffset(position);\n    };\n\n    if (useScrollEvent) {\n      window.addEventListener('scroll', handleScroll, { passive: true });\n    } else {\n      window.removeEventListener('scroll', handleScroll);\n    }\n\n    return () => {\n      window.removeEventListener('scroll', handleScroll);\n    };\n  }, [useScrollEvent, startOffset]);\n\n  const checkScreenSize = () => {\n    if (window.innerWidth >= 1301) {\n      // Wide. If larger, and there is a startOffset, the breadcrumb is positioned absolute at start\n      setUseScrollEvent(true);\n    } else {\n      setUseScrollEvent(false);\n    }\n  };\n\n  useEffect(() => {\n    checkScreenSize();\n    window.addEventListener('resize', checkScreenSize);\n\n    return () => {\n      window.removeEventListener('resize', checkScreenSize);\n    };\n  }, []);\n\n  return (\n    <>\n      <Wrapper\n        leftAlign={leftAlign}\n        startOffset={wrapperOffset}\n        hideOnNarrow={hideOnNarrow}\n        isVisible={isVisible}\n        mastheadHeight={mastheadHeight}>\n        {items.length > 0 && (\n          <>\n            <Heading invertedStyle={invertedStyle}>{t('breadcrumb.youAreHere')}</Heading>\n            {messageBoxTagMessage && <MessageBoxTag tagMessage={messageBoxTagMessage} />}\n            <List data-testid=\"breadcrumb-list\">\n              {items.map((item: BreadcrumbItemProps) => {\n                const { id, label, url, typename, icon, isCurrent = false } = item;\n                return (\n                  <ListItem invertedStyle={invertedStyle} key={`${id}-${typename}`}>\n                    {isCurrent && <Dot invertedStyle={invertedStyle} />}\n                    <SafeLink\n                      className=\"linkitem\"\n                      to={url}\n                      onClick={(e: MouseEvent<HTMLElement>) => {\n                        onNav && onNav(e, item);\n                      }}\n                      aria-label={label}>\n                      <IconWrapper invertedStyle={invertedStyle} isCurrent={isCurrent}>\n                        {icon && icon}\n                        {typename && TypeIcon(typename)}\n                      </IconWrapper>\n                      <span>{label}</span>\n                    </SafeLink>\n                  </ListItem>\n                );\n              })}\n            </List>\n          </>\n        )}\n        {children}\n      </Wrapper>\n    </>\n  );\n};\n\nexport default Breadcrumblist;\n"]} */");
112
112
 
113
113
  var TypeIcon = function TypeIcon(type) {
114
114
  switch (type) {
@@ -1,2 +1,3 @@
1
1
  import Breadcrumblist from './Breadcrumblist';
2
+ export type { BreadcrumbItemProps } from './Breadcrumblist';
2
3
  export default Breadcrumblist;