@ndla/ui 18.0.0 → 18.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/es/Masthead/MastheadAuthModal.js +8 -3
  2. package/es/MyNdla/Resource/Folder.js +7 -7
  3. package/es/Resource/resourceComponents.js +8 -8
  4. package/es/TreeStructure/FolderItems.js +3 -3
  5. package/es/TreeStructure/TreeStructure.js +51 -73
  6. package/es/TreeStructure/keyboardNavigation/keyboardNavigation.js +23 -11
  7. package/es/locale/messages-en.js +1 -0
  8. package/es/locale/messages-nb.js +1 -0
  9. package/es/locale/messages-nn.js +1 -0
  10. package/es/locale/messages-se.js +1 -0
  11. package/es/locale/messages-sma.js +1 -0
  12. package/lib/Masthead/MastheadAuthModal.js +14 -7
  13. package/lib/MyNdla/Resource/Folder.js +7 -7
  14. package/lib/Resource/resourceComponents.js +8 -8
  15. package/lib/TreeStructure/FolderItems.js +3 -3
  16. package/lib/TreeStructure/TreeStructure.js +52 -73
  17. package/lib/TreeStructure/TreeStructure.types.d.ts +2 -2
  18. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.js +23 -11
  19. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.types.d.ts +1 -1
  20. package/lib/locale/messages-en.d.ts +1 -0
  21. package/lib/locale/messages-en.js +1 -0
  22. package/lib/locale/messages-nb.d.ts +1 -0
  23. package/lib/locale/messages-nb.js +1 -0
  24. package/lib/locale/messages-nn.d.ts +1 -0
  25. package/lib/locale/messages-nn.js +1 -0
  26. package/lib/locale/messages-se.d.ts +1 -0
  27. package/lib/locale/messages-se.js +1 -0
  28. package/lib/locale/messages-sma.d.ts +1 -0
  29. package/lib/locale/messages-sma.js +1 -0
  30. package/package.json +2 -2
  31. package/src/Masthead/MastheadAuthModal.tsx +9 -0
  32. package/src/MyNdla/Resource/Folder.tsx +1 -1
  33. package/src/Resource/resourceComponents.tsx +3 -0
  34. package/src/TreeStructure/FolderItems.tsx +1 -1
  35. package/src/TreeStructure/TreeStructure.tsx +33 -35
  36. package/src/TreeStructure/TreeStructure.types.ts +2 -2
  37. package/src/TreeStructure/keyboardNavigation/keyboardNavigation.ts +7 -7
  38. package/src/TreeStructure/keyboardNavigation/keyboardNavigation.types.ts +1 -1
  39. package/src/locale/messages-en.ts +1 -0
  40. package/src/locale/messages-nb.ts +1 -0
  41. package/src/locale/messages-nn.ts +1 -0
  42. package/src/locale/messages-se.ts +1 -0
  43. package/src/locale/messages-sma.ts +1 -0
@@ -21,6 +21,8 @@ var _reactI18next = require("react-i18next");
21
21
 
22
22
  var _core = require("@ndla/core");
23
23
 
24
+ var _lodash = require("lodash");
25
+
24
26
  var _TreeStructureWrapper = _interopRequireDefault(require("./TreeStructureWrapper"));
25
27
 
26
28
  var _FolderItems = _interopRequireDefault(require("./FolderItems"));
@@ -43,9 +45,13 @@ function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { va
43
45
 
44
46
  function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
45
47
 
46
- function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
48
+ function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); }
49
+
50
+ function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
47
51
 
48
- function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
52
+ function _iterableToArray(iter) { if (typeof Symbol !== "undefined" && Symbol.iterator in Object(iter)) return Array.from(iter); }
53
+
54
+ function _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); }
49
55
 
50
56
  function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
51
57
 
@@ -64,13 +70,15 @@ exports.MAX_LEVEL_FOR_FOLDERS = MAX_LEVEL_FOR_FOLDERS;
64
70
  var StyledLabel = (0, _styledBase["default"])("label", {
65
71
  target: "e1dg1gdn0",
66
72
  label: "StyledLabel"
67
- })("font-weight:", _core.fonts.weight.semibold, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAuBgC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<Set<string>>(new Set(defaultOpenFolders || []));\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0].id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    setOpenFolders((prev) => {\n      defaultOpenFolders?.forEach((id) => prev.add(id));\n      return new Set(prev);\n    });\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    setOpenFolders((prev) => {\n      if (prev.has(id)) {\n        prev.delete(id);\n        // Did we just closed a folder with a marked folder inside it?\n        // If so, we need to mark the folder we just closed.\n        if (markedFolderId) {\n          const closingFolderPath = getPathOfFolder(data, id);\n          const markedFolderPath = getPathOfFolder(data, markedFolderId);\n          const markedFolderIsSubPath = closingFolderPath.every(\n            (folderId, _index) => markedFolderPath[_index] === folderId,\n          );\n          if (markedFolderIsSubPath) {\n            setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n          }\n        }\n      } else {\n        prev.add(id);\n      }\n      return new Set(prev);\n    });\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = async (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      const newFolderId = await onNewFolder({ ...newFolder, value });\n      if (newFolderId) {\n        setMarkedFolderId(newFolderId);\n        setFocusedFolderId(newFolderId);\n        // Open current folder in case it was closed..\n        setOpenFolders((prev) => {\n          if (newFolder.parentId) {\n            prev.add(newFolder.parentId);\n          }\n          return new Set(prev);\n        });\n      }\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <AddButton\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}\n            />\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
73
+ })("font-weight:", _core.fonts.weight.semibold, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAwBgC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport { uniq } from 'lodash';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0]?.id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      setOpenFolders((prev) => {\n        return uniq([...defaultOpenFolders, ...prev]);\n      });\n    }\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    if (openFolders.includes(id)) {\n      // Did we just closed a folder with a marked folder inside it?\n      // If so, we need to mark the folder we just closed.\n      if (markedFolderId) {\n        const closingFolderPath = getPathOfFolder(data, id);\n        const markedFolderPath = getPathOfFolder(data, markedFolderId);\n        const markedFolderIsSubPath = closingFolderPath.every(\n          (folderId, _index) => markedFolderPath[_index] === folderId,\n        );\n        if (markedFolderIsSubPath) {\n          setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n        }\n      }\n      setOpenFolders(openFolders.filter((folder) => folder !== id));\n    } else {\n      setOpenFolders(uniq([...openFolders, id]));\n    }\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      onNewFolder({ ...newFolder, value }).then((newFolderId) => {\n        if (newFolderId) {\n          setMarkedFolderId(newFolderId);\n          setFocusedFolderId(newFolderId);\n          // Open current folder in case it was closed..\n\n          if (newFolder.parentId) {\n            setOpenFolders(uniq([...openFolders, newFolder.parentId]));\n          }\n        }\n      });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <AddButton\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}\n            />\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
68
74
  var AddFolderWrapper = (0, _styledBase["default"])("div", {
69
75
  target: "e1dg1gdn1",
70
76
  label: "AddFolderWrapper"
71
- })("display:flex;margin-top:", _core.spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA2BmC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<Set<string>>(new Set(defaultOpenFolders || []));\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0].id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    setOpenFolders((prev) => {\n      defaultOpenFolders?.forEach((id) => prev.add(id));\n      return new Set(prev);\n    });\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    setOpenFolders((prev) => {\n      if (prev.has(id)) {\n        prev.delete(id);\n        // Did we just closed a folder with a marked folder inside it?\n        // If so, we need to mark the folder we just closed.\n        if (markedFolderId) {\n          const closingFolderPath = getPathOfFolder(data, id);\n          const markedFolderPath = getPathOfFolder(data, markedFolderId);\n          const markedFolderIsSubPath = closingFolderPath.every(\n            (folderId, _index) => markedFolderPath[_index] === folderId,\n          );\n          if (markedFolderIsSubPath) {\n            setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n          }\n        }\n      } else {\n        prev.add(id);\n      }\n      return new Set(prev);\n    });\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = async (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      const newFolderId = await onNewFolder({ ...newFolder, value });\n      if (newFolderId) {\n        setMarkedFolderId(newFolderId);\n        setFocusedFolderId(newFolderId);\n        // Open current folder in case it was closed..\n        setOpenFolders((prev) => {\n          if (newFolder.parentId) {\n            prev.add(newFolder.parentId);\n          }\n          return new Set(prev);\n        });\n      }\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <AddButton\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}\n            />\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
77
+ })("display:flex;margin-top:", _core.spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA4BmC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport { uniq } from 'lodash';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0]?.id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      setOpenFolders((prev) => {\n        return uniq([...defaultOpenFolders, ...prev]);\n      });\n    }\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    if (openFolders.includes(id)) {\n      // Did we just closed a folder with a marked folder inside it?\n      // If so, we need to mark the folder we just closed.\n      if (markedFolderId) {\n        const closingFolderPath = getPathOfFolder(data, id);\n        const markedFolderPath = getPathOfFolder(data, markedFolderId);\n        const markedFolderIsSubPath = closingFolderPath.every(\n          (folderId, _index) => markedFolderPath[_index] === folderId,\n        );\n        if (markedFolderIsSubPath) {\n          setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n        }\n      }\n      setOpenFolders(openFolders.filter((folder) => folder !== id));\n    } else {\n      setOpenFolders(uniq([...openFolders, id]));\n    }\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      onNewFolder({ ...newFolder, value }).then((newFolderId) => {\n        if (newFolderId) {\n          setMarkedFolderId(newFolderId);\n          setFocusedFolderId(newFolderId);\n          // Open current folder in case it was closed..\n\n          if (newFolder.parentId) {\n            setOpenFolders(uniq([...openFolders, newFolder.parentId]));\n          }\n        }\n      });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <AddButton\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}\n            />\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
72
78
 
73
79
  var TreeStructure = function TreeStructure(_ref) {
80
+ var _data$;
81
+
74
82
  var data = _ref.data,
75
83
  label = _ref.label,
76
84
  editable = _ref.editable,
@@ -90,7 +98,7 @@ var TreeStructure = function TreeStructure(_ref) {
90
98
  newFolder = _useState2[0],
91
99
  setNewFolder = _useState2[1];
92
100
 
93
- var _useState3 = (0, _react.useState)(new Set(defaultOpenFolders || [])),
101
+ var _useState3 = (0, _react.useState)(defaultOpenFolders || []),
94
102
  _useState4 = _slicedToArray(_useState3, 2),
95
103
  openFolders = _useState4[0],
96
104
  setOpenFolders = _useState4[1];
@@ -100,7 +108,7 @@ var TreeStructure = function TreeStructure(_ref) {
100
108
  focusedFolderId = _useState6[0],
101
109
  setFocusedFolderId = _useState6[1];
102
110
 
103
- var _useState7 = (0, _react.useState)(folderIdMarkedByDefault || data[0].id),
111
+ var _useState7 = (0, _react.useState)(folderIdMarkedByDefault || ((_data$ = data[0]) === null || _data$ === void 0 ? void 0 : _data$.id)),
104
112
  _useState8 = _slicedToArray(_useState7, 2),
105
113
  markedFolderId = _useState8[0],
106
114
  setMarkedFolderId = _useState8[1];
@@ -112,12 +120,11 @@ var TreeStructure = function TreeStructure(_ref) {
112
120
  }, []); // TODO: use useId hook when we update to React 18
113
121
 
114
122
  (0, _react.useEffect)(function () {
115
- setOpenFolders(function (prev) {
116
- defaultOpenFolders === null || defaultOpenFolders === void 0 ? void 0 : defaultOpenFolders.forEach(function (id) {
117
- return prev.add(id);
123
+ if (defaultOpenFolders) {
124
+ setOpenFolders(function (prev) {
125
+ return (0, _lodash.uniq)([].concat(_toConsumableArray(defaultOpenFolders), _toConsumableArray(prev)));
118
126
  });
119
- return new Set(prev);
120
- });
127
+ }
121
128
  }, [defaultOpenFolders]);
122
129
  (0, _react.useEffect)(function () {
123
130
  if (!loading) {
@@ -126,79 +133,51 @@ var TreeStructure = function TreeStructure(_ref) {
126
133
  }, [loading]);
127
134
 
128
135
  var onToggleOpen = function onToggleOpen(id) {
129
- setOpenFolders(function (prev) {
130
- if (prev.has(id)) {
131
- prev["delete"](id); // Did we just closed a folder with a marked folder inside it?
132
- // If so, we need to mark the folder we just closed.
133
-
134
- if (markedFolderId) {
135
- var closingFolderPath = (0, _helperFunctions.getPathOfFolder)(data, id);
136
- var markedFolderPath = (0, _helperFunctions.getPathOfFolder)(data, markedFolderId);
137
- var markedFolderIsSubPath = closingFolderPath.every(function (folderId, _index) {
138
- return markedFolderPath[_index] === folderId;
139
- });
140
-
141
- if (markedFolderIsSubPath) {
142
- setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);
143
- }
136
+ if (openFolders.includes(id)) {
137
+ // Did we just closed a folder with a marked folder inside it?
138
+ // If so, we need to mark the folder we just closed.
139
+ if (markedFolderId) {
140
+ var closingFolderPath = (0, _helperFunctions.getPathOfFolder)(data, id);
141
+ var markedFolderPath = (0, _helperFunctions.getPathOfFolder)(data, markedFolderId);
142
+ var markedFolderIsSubPath = closingFolderPath.every(function (folderId, _index) {
143
+ return markedFolderPath[_index] === folderId;
144
+ });
145
+
146
+ if (markedFolderIsSubPath) {
147
+ setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);
144
148
  }
145
- } else {
146
- prev.add(id);
147
149
  }
148
150
 
149
- return new Set(prev);
150
- });
151
+ setOpenFolders(openFolders.filter(function (folder) {
152
+ return folder !== id;
153
+ }));
154
+ } else {
155
+ setOpenFolders((0, _lodash.uniq)([].concat(_toConsumableArray(openFolders), [id])));
156
+ }
151
157
  };
152
158
 
153
159
  var onCreateNewFolder = function onCreateNewFolder(props) {
154
160
  setNewFolder(props);
155
161
  };
156
162
 
157
- var onSaveNewFolder = /*#__PURE__*/function () {
158
- var _ref2 = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee(value) {
159
- var newFolderId;
160
- return regeneratorRuntime.wrap(function _callee$(_context) {
161
- while (1) {
162
- switch (_context.prev = _context.next) {
163
- case 0:
164
- if (!newFolder) {
165
- _context.next = 5;
166
- break;
167
- }
168
-
169
- _context.next = 3;
170
- return onNewFolder(_objectSpread(_objectSpread({}, newFolder), {}, {
171
- value: value
172
- }));
173
-
174
- case 3:
175
- newFolderId = _context.sent;
176
-
177
- if (newFolderId) {
178
- setMarkedFolderId(newFolderId);
179
- setFocusedFolderId(newFolderId); // Open current folder in case it was closed..
180
-
181
- setOpenFolders(function (prev) {
182
- if (newFolder.parentId) {
183
- prev.add(newFolder.parentId);
184
- }
185
-
186
- return new Set(prev);
187
- });
188
- }
189
-
190
- case 5:
191
- case "end":
192
- return _context.stop();
163
+ var onSaveNewFolder = function onSaveNewFolder(value) {
164
+ if (newFolder) {
165
+ // We would like to create a new folder with the name of value.
166
+ // Its location in structure is based on newFolder object
167
+ onNewFolder(_objectSpread(_objectSpread({}, newFolder), {}, {
168
+ value: value
169
+ })).then(function (newFolderId) {
170
+ if (newFolderId) {
171
+ setMarkedFolderId(newFolderId);
172
+ setFocusedFolderId(newFolderId); // Open current folder in case it was closed..
173
+
174
+ if (newFolder.parentId) {
175
+ setOpenFolders((0, _lodash.uniq)([].concat(_toConsumableArray(openFolders), [newFolder.parentId])));
193
176
  }
194
177
  }
195
- }, _callee);
196
- }));
197
-
198
- return function onSaveNewFolder(_x) {
199
- return _ref2.apply(this, arguments);
200
- };
201
- }();
178
+ });
179
+ }
180
+ };
202
181
 
203
182
  var onCancelNewFolder = function onCancelNewFolder() {
204
183
  setNewFolder(undefined);
@@ -43,7 +43,7 @@ export declare type onCreateNewFolderProp = ({ idPaths, parentId, }: {
43
43
  idPaths: number[];
44
44
  parentId: string | undefined;
45
45
  }) => void;
46
- export declare type SetOpenFolderProp = React.Dispatch<React.SetStateAction<Set<string>>>;
46
+ export declare type SetOpenFolderProp = React.Dispatch<React.SetStateAction<string[]>>;
47
47
  export declare type SetFocusedFolderId = React.Dispatch<React.SetStateAction<string | undefined>>;
48
48
  export declare type FolderChildFuncType = (id: string, tabIndex: number) => ReactNode;
49
49
  export interface FolderItemsProps extends CommonFolderProps {
@@ -52,7 +52,7 @@ export interface FolderItemsProps extends CommonFolderProps {
52
52
  onCancelNewFolder: () => void;
53
53
  onCreateNewFolder: onCreateNewFolderProp;
54
54
  newFolder: NewFolderProps | undefined;
55
- openFolders: Set<string>;
55
+ openFolders: string[];
56
56
  markedFolderId?: string;
57
57
  onMarkFolder: (id: string) => void;
58
58
  idPaths: number[];
@@ -50,7 +50,7 @@ var traverseUpwards = function traverseUpwards(inital, setFocusedFolderId, paths
50
50
  };
51
51
 
52
52
  var keyboardNavigation = function keyboardNavigation(_ref) {
53
- var _document$activeEleme, _elementWithKeyFocus$3;
53
+ var _document$activeEleme, _elementWithKeyFocus$5;
54
54
 
55
55
  var e = _ref.e,
56
56
  data = _ref.data,
@@ -79,7 +79,7 @@ var keyboardNavigation = function keyboardNavigation(_ref) {
79
79
  if (dataId === id) {
80
80
  elementWithKeyFocus.paths = paths;
81
81
  elementWithKeyFocus.index = _index;
82
- elementWithKeyFocus.isOpen = openFolders.has(dataId) && childData && childData.length > 0;
82
+ elementWithKeyFocus.isOpen = openFolders.includes(dataId) && childData && childData.length > 0;
83
83
  elementWithKeyFocus.data = childData;
84
84
  elementWithKeyFocus.parent = parent;
85
85
  elementWithKeyFocus.parentId = parentId;
@@ -92,9 +92,11 @@ var keyboardNavigation = function keyboardNavigation(_ref) {
92
92
  };
93
93
 
94
94
  if (!updatePathToElementWithKeyFocus(data, [], data)) {
95
+ var _data$;
96
+
95
97
  // Couldn't find its location in the tree.
96
98
  // This should not happen, reset its value to root.
97
- setFocusedFolderId(e.key === 'ArrowDown' ? data[0].id : undefined);
99
+ setFocusedFolderId(e.key === 'ArrowDown' ? (_data$ = data[0]) === null || _data$ === void 0 ? void 0 : _data$.id : undefined);
98
100
  return;
99
101
  }
100
102
 
@@ -130,7 +132,9 @@ var keyboardNavigation = function keyboardNavigation(_ref) {
130
132
  }
131
133
 
132
134
  if (!id && e.key === 'ArrowDown') {
133
- setFocusedFolderId(data[0].id);
135
+ var _data$2;
136
+
137
+ setFocusedFolderId((_data$2 = data[0]) === null || _data$2 === void 0 ? void 0 : _data$2.id);
134
138
  return;
135
139
  }
136
140
 
@@ -141,9 +145,13 @@ var keyboardNavigation = function keyboardNavigation(_ref) {
141
145
 
142
146
  if (e.key === 'ArrowUp') {
143
147
  if (elementWithKeyFocus.index > 0) {
148
+ var _elementWithKeyFocus$2;
149
+
144
150
  // Move upwards to the parent folder
145
- setFocusedFolderId(elementWithKeyFocus.parent ? elementWithKeyFocus.parent[elementWithKeyFocus.index - 1].id : undefined);
151
+ setFocusedFolderId(elementWithKeyFocus.parent ? (_elementWithKeyFocus$2 = elementWithKeyFocus.parent[elementWithKeyFocus.index - 1]) === null || _elementWithKeyFocus$2 === void 0 ? void 0 : _elementWithKeyFocus$2.id : undefined);
146
152
  } else if (elementWithKeyFocus.paths.length > 0) {
153
+ var _findParent$parentsCu;
154
+
147
155
  elementWithKeyFocus.paths.pop();
148
156
  var findParent = data;
149
157
  elementWithKeyFocus.paths.forEach(function (index) {
@@ -153,17 +161,19 @@ var keyboardNavigation = function keyboardNavigation(_ref) {
153
161
  var id = _ref3.id;
154
162
  return id === elementWithKeyFocus.parentId;
155
163
  });
156
- setFocusedFolderId(findParent[parentsCurrentIndex].id);
164
+ setFocusedFolderId((_findParent$parentsCu = findParent[parentsCurrentIndex]) === null || _findParent$parentsCu === void 0 ? void 0 : _findParent$parentsCu.id);
157
165
  }
158
166
 
159
167
  return;
160
168
  }
161
169
 
162
170
  if (elementWithKeyFocus.isOpen) {
163
- var _elementWithKeyFocus$2;
171
+ var _elementWithKeyFocus$3;
172
+
173
+ if ((_elementWithKeyFocus$3 = elementWithKeyFocus.data) === null || _elementWithKeyFocus$3 === void 0 ? void 0 : _elementWithKeyFocus$3.length) {
174
+ var _elementWithKeyFocus$4;
164
175
 
165
- if ((_elementWithKeyFocus$2 = elementWithKeyFocus.data) === null || _elementWithKeyFocus$2 === void 0 ? void 0 : _elementWithKeyFocus$2.length) {
166
- setFocusedFolderId(elementWithKeyFocus.data[0].id);
176
+ setFocusedFolderId((_elementWithKeyFocus$4 = elementWithKeyFocus.data[0]) === null || _elementWithKeyFocus$4 === void 0 ? void 0 : _elementWithKeyFocus$4.id);
167
177
  } else {
168
178
  // move to next child of parent if any... need new traverse :-/
169
179
  traverseUpwards(data, setFocusedFolderId, elementWithKeyFocus.paths, elementWithKeyFocus.index);
@@ -172,9 +182,11 @@ var keyboardNavigation = function keyboardNavigation(_ref) {
172
182
  return;
173
183
  }
174
184
 
175
- if (elementWithKeyFocus.parent && elementWithKeyFocus.index < ((_elementWithKeyFocus$3 = elementWithKeyFocus.parent) === null || _elementWithKeyFocus$3 === void 0 ? void 0 : _elementWithKeyFocus$3.length) - 1) {
185
+ if (elementWithKeyFocus.parent && elementWithKeyFocus.index < ((_elementWithKeyFocus$5 = elementWithKeyFocus.parent) === null || _elementWithKeyFocus$5 === void 0 ? void 0 : _elementWithKeyFocus$5.length) - 1) {
186
+ var _elementWithKeyFocus$6;
187
+
176
188
  // Move downwards to the next child
177
- setFocusedFolderId(elementWithKeyFocus.parent[elementWithKeyFocus.index + 1].id);
189
+ setFocusedFolderId((_elementWithKeyFocus$6 = elementWithKeyFocus.parent[elementWithKeyFocus.index + 1]) === null || _elementWithKeyFocus$6 === void 0 ? void 0 : _elementWithKeyFocus$6.id);
178
190
  return;
179
191
  }
180
192
 
@@ -11,7 +11,7 @@ export interface KeyboardNavigationProps {
11
11
  e: React.KeyboardEvent<HTMLElement>;
12
12
  data: FolderStructureProps[];
13
13
  setFocusedFolderId: SetFocusedFolderId;
14
- openFolders: Set<string>;
14
+ openFolders: string[];
15
15
  onToggleOpen: (id: string) => void;
16
16
  focusedFolderId: string | undefined;
17
17
  }
@@ -53,6 +53,7 @@ declare const messages: {
53
53
  close: string;
54
54
  };
55
55
  myNdla: {
56
+ myNDLA: string;
56
57
  resources: string;
57
58
  resources_plural: string;
58
59
  folders: string;
@@ -945,6 +945,7 @@ var messages = _objectSpread(_objectSpread({
945
945
  close: 'Close fact box'
946
946
  },
947
947
  myNdla: {
948
+ myNDLA: 'My NDLA',
948
949
  resources: '{{count}} Resource',
949
950
  resources_plural: '{{count}} Resources',
950
951
  folders: '{{count}} Folder',
@@ -53,6 +53,7 @@ declare const messages: {
53
53
  close: string;
54
54
  };
55
55
  myNdla: {
56
+ myNDLA: string;
56
57
  resources: string;
57
58
  resources_plural: string;
58
59
  folders: string;
@@ -945,6 +945,7 @@ var messages = _objectSpread(_objectSpread({
945
945
  close: 'Lukk faktaboks'
946
946
  },
947
947
  myNdla: {
948
+ myNDLA: 'Min NDLA',
948
949
  resources: '{{count}} ressurs',
949
950
  resources_plural: '{{count}} ressurser',
950
951
  folders: '{{count}} mappe',
@@ -53,6 +53,7 @@ declare const messages: {
53
53
  close: string;
54
54
  };
55
55
  myNdla: {
56
+ myNDLA: string;
56
57
  resources: string;
57
58
  resources_plural: string;
58
59
  folders: string;
@@ -945,6 +945,7 @@ var messages = _objectSpread(_objectSpread({
945
945
  close: 'Lukk faktaboks'
946
946
  },
947
947
  myNdla: {
948
+ myNDLA: 'Min NDLA',
948
949
  resources: '{{count}} ressurs',
949
950
  resources_plural: '{{count}} ressursar',
950
951
  folders: '{{count}} mappe',
@@ -53,6 +53,7 @@ declare const messages: {
53
53
  close: string;
54
54
  };
55
55
  myNdla: {
56
+ myNDLA: string;
56
57
  resources: string;
57
58
  resources_plural: string;
58
59
  folders: string;
@@ -945,6 +945,7 @@ var messages = _objectSpread(_objectSpread({
945
945
  close: 'Lukk faktaboks'
946
946
  },
947
947
  myNdla: {
948
+ myNDLA: 'Min NDLA',
948
949
  resources: '{{count}} ressurs',
949
950
  resources_plural: '{{count}} ressurser',
950
951
  folders: '{{count}} mappe',
@@ -53,6 +53,7 @@ declare const messages: {
53
53
  close: string;
54
54
  };
55
55
  myNdla: {
56
+ myNDLA: string;
56
57
  resources: string;
57
58
  resources_plural: string;
58
59
  folders: string;
@@ -945,6 +945,7 @@ var messages = _objectSpread(_objectSpread({
945
945
  close: 'Lukk faktaboks'
946
946
  },
947
947
  myNdla: {
948
+ myNDLA: 'Min NDLA',
948
949
  resources: '{{count}} ressurs',
949
950
  resources_plural: '{{count}} ressurser',
950
951
  folders: '{{count}} mappe',
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ndla/ui",
3
- "version": "18.0.0",
3
+ "version": "18.0.1",
4
4
  "description": "UI component library for NDLA.",
5
5
  "license": "GPL-3.0",
6
6
  "main": "lib/index.js",
@@ -81,5 +81,5 @@
81
81
  "publishConfig": {
82
82
  "access": "public"
83
83
  },
84
- "gitHead": "917673dcc652e2cbc9106b8b4c30cd393c00cff5"
84
+ "gitHead": "841ff68015ec6032bed44a7c25eea2dfd9088a64"
85
85
  }
@@ -10,6 +10,8 @@ import React from 'react';
10
10
  import styled from '@emotion/styled';
11
11
  import Button from '@ndla/button';
12
12
  import { Feide } from '@ndla/icons/common';
13
+ import { colors, fonts, spacing } from '@ndla/core';
14
+ import { useTranslation } from 'react-i18next';
13
15
 
14
16
  import AuthModal, { AuthModalProps } from '../User/AuthModal';
15
17
 
@@ -18,7 +20,12 @@ type FeideWrapperProps = {
18
20
  };
19
21
 
20
22
  const StyledButton = styled(Button)<FeideWrapperProps>`
23
+ font-weight: ${fonts.weight.semibold};
24
+ display: flex;
25
+ align-items: center;
21
26
  .feide-icon svg {
27
+ margin-left: ${spacing.xsmall};
28
+ fill: ${colors.brand.primary};
22
29
  color: ${(props) => (props.inverted ? `#ffffff` : `#000000`)};
23
30
  width: 22px;
24
31
  height: 22px;
@@ -35,11 +42,13 @@ interface Props extends AuthModalProps {
35
42
  }
36
43
 
37
44
  const MastheadAuthModal = ({ inverted, ...rest }: Props) => {
45
+ const { t } = useTranslation();
38
46
  return (
39
47
  <AuthModal
40
48
  {...rest}
41
49
  activateButton={
42
50
  <StyledButton inverted={inverted} ghostPill={!inverted} ghostPillInverted={inverted} aria-label="Feide">
51
+ {t('myNdla.myNDLA')}
43
52
  <span className="feide-icon">
44
53
  <Feide />
45
54
  </span>
@@ -124,7 +124,7 @@ const IconCount = ({ type, count, layoutType }: IconCountProps) => {
124
124
  return (
125
125
  <IconCountWrapper type={layoutType}>
126
126
  <Icon aria-label={t(`myNdla.${type}s`)} />
127
- <span>{layoutType === 'block' ? count : t(`myNdla.${type}s`, { count: 3 })}</span>
127
+ <span>{layoutType === 'block' ? count : t(`myNdla.${type}s`, { count })}</span>
128
128
  </IconCountWrapper>
129
129
  );
130
130
  };
@@ -48,6 +48,9 @@ const StyledTagListElement = styled.li`
48
48
  ::before {
49
49
  content: '#';
50
50
  }
51
+ &:hover {
52
+ color: ${colors.brand.primary};
53
+ }
51
54
  `;
52
55
 
53
56
  const StyledTopicList = styled.ul`
@@ -53,7 +53,7 @@ const FolderItems = ({
53
53
  <StyledUL role="group" firstLevel={firstLevel}>
54
54
  {data.map(({ name, data: dataChildren, id, url, icon }, _index) => {
55
55
  const newIdPaths = [...idPaths, _index];
56
- const isOpen = openFolders?.has(id);
56
+ const isOpen = openFolders?.includes(id);
57
57
  return (
58
58
  <StyledLI key={id} role="treeitem">
59
59
  <div>