@ndla/ui 25.2.1 → 26.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 (74) hide show
  1. package/es/Article/ArticleByline.js +17 -7
  2. package/es/Article/ArticleSideBar.js +5 -4
  3. package/es/Breadcrumb/BreadcrumbItem.js +8 -7
  4. package/es/ErrorMessage/ErrorMessage.js +12 -6
  5. package/es/Frontpage/FrontpageHeader.js +7 -9
  6. package/es/LanguageSelector/LanguageSelector.js +12 -7
  7. package/es/LearningPaths/LearningPathInformation.js +8 -5
  8. package/es/Subject/SubjectHeader.js +5 -6
  9. package/es/TreeStructure/FolderItem.js +110 -94
  10. package/es/TreeStructure/FolderItems.js +26 -30
  11. package/es/TreeStructure/FolderNameInput.js +35 -27
  12. package/es/TreeStructure/NavigationLink.js +81 -0
  13. package/es/TreeStructure/TreeStructure.js +169 -45
  14. package/es/locale/messages-en.js +7 -22
  15. package/es/locale/messages-nb.js +8 -23
  16. package/es/locale/messages-nn.js +7 -22
  17. package/es/locale/messages-se.js +697 -712
  18. package/es/locale/messages-sma.js +8 -23
  19. package/lib/Article/ArticleByline.js +17 -7
  20. package/lib/Article/ArticleSideBar.js +5 -4
  21. package/lib/Breadcrumb/BreadcrumbItem.js +8 -7
  22. package/lib/ErrorMessage/ErrorMessage.d.ts +1 -0
  23. package/lib/ErrorMessage/ErrorMessage.js +12 -6
  24. package/lib/Frontpage/FrontpageHeader.d.ts +5 -6
  25. package/lib/Frontpage/FrontpageHeader.js +7 -11
  26. package/lib/LanguageSelector/LanguageSelector.js +13 -7
  27. package/lib/LearningPaths/LearningPathInformation.d.ts +2 -1
  28. package/lib/LearningPaths/LearningPathInformation.js +8 -5
  29. package/lib/Subject/SubjectHeader.js +14 -16
  30. package/lib/TreeStructure/FolderItem.d.ts +2 -3
  31. package/lib/TreeStructure/FolderItem.js +107 -92
  32. package/lib/TreeStructure/FolderItems.d.ts +1 -3
  33. package/lib/TreeStructure/FolderItems.js +26 -29
  34. package/lib/TreeStructure/FolderNameInput.d.ts +2 -1
  35. package/lib/TreeStructure/FolderNameInput.js +33 -26
  36. package/lib/TreeStructure/NavigationLink.d.ts +15 -0
  37. package/lib/TreeStructure/NavigationLink.js +100 -0
  38. package/lib/TreeStructure/TreeStructure.d.ts +1 -2
  39. package/lib/TreeStructure/TreeStructure.js +163 -45
  40. package/lib/TreeStructure/types.d.ts +4 -1
  41. package/lib/locale/messages-en.d.ts +4 -19
  42. package/lib/locale/messages-en.js +7 -22
  43. package/lib/locale/messages-nb.d.ts +4 -19
  44. package/lib/locale/messages-nb.js +8 -23
  45. package/lib/locale/messages-nn.d.ts +4 -19
  46. package/lib/locale/messages-nn.js +7 -22
  47. package/lib/locale/messages-se.d.ts +4 -19
  48. package/lib/locale/messages-se.js +697 -712
  49. package/lib/locale/messages-sma.d.ts +4 -19
  50. package/lib/locale/messages-sma.js +8 -23
  51. package/package.json +14 -14
  52. package/src/Article/ArticleByline.tsx +10 -3
  53. package/src/Article/ArticleSideBar.tsx +1 -0
  54. package/src/Breadcrumb/BreadcrumbItem.tsx +1 -1
  55. package/src/ErrorMessage/ErrorMessage.tsx +6 -0
  56. package/src/Frontpage/FrontpageHeader.tsx +5 -6
  57. package/src/LanguageSelector/LanguageSelector.tsx +4 -1
  58. package/src/LearningPaths/LearningPathInformation.tsx +3 -2
  59. package/src/Subject/SubjectHeader.tsx +1 -2
  60. package/src/TreeStructure/FolderItem.tsx +126 -104
  61. package/src/TreeStructure/FolderItems.tsx +51 -43
  62. package/src/TreeStructure/FolderNameInput.tsx +43 -28
  63. package/src/TreeStructure/NavigationLink.tsx +100 -0
  64. package/src/TreeStructure/TreeStructure.tsx +187 -61
  65. package/src/TreeStructure/types.ts +5 -1
  66. package/src/locale/messages-en.ts +9 -22
  67. package/src/locale/messages-nb.ts +10 -23
  68. package/src/locale/messages-nn.ts +9 -22
  69. package/src/locale/messages-se.ts +724 -738
  70. package/src/locale/messages-sma.ts +10 -23
  71. package/es/TreeStructure/TreeStructureWrapper.js +0 -13
  72. package/lib/TreeStructure/TreeStructureWrapper.d.ts +0 -12
  73. package/lib/TreeStructure/TreeStructureWrapper.js +0 -24
  74. package/src/TreeStructure/TreeStructureWrapper.tsx +0 -31
@@ -13,22 +13,24 @@ var _react = _interopRequireWildcard(require("react"));
13
13
 
14
14
  var _button = require("@ndla/button");
15
15
 
16
+ var _action = require("@ndla/icons/action");
17
+
18
+ var _common = require("@ndla/icons/common");
19
+
16
20
  var _tooltip = _interopRequireDefault(require("@ndla/tooltip"));
17
21
 
18
22
  var _reactI18next = require("react-i18next");
19
23
 
20
24
  var _core = require("@ndla/core");
21
25
 
22
- var _lodash = require("lodash");
26
+ var _core2 = require("@emotion/core");
23
27
 
24
- var _TreeStructureWrapper = _interopRequireDefault(require("./TreeStructureWrapper"));
28
+ var _lodash = require("lodash");
25
29
 
26
30
  var _FolderItems = _interopRequireDefault(require("./FolderItems"));
27
31
 
28
32
  var _helperFunctions = require("./helperFunctions");
29
33
 
30
- var _core2 = require("@emotion/core");
31
-
32
34
  function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; }
33
35
 
34
36
  function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
@@ -47,35 +49,102 @@ function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !(
47
49
 
48
50
  function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
49
51
 
52
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
53
+
50
54
  var MAX_LEVEL_FOR_FOLDERS = 4;
51
55
  exports.MAX_LEVEL_FOR_FOLDERS = MAX_LEVEL_FOR_FOLDERS;
52
56
  var StyledLabel = (0, _styledBase["default"])("label", {
53
57
  target: "e1dg1gdn0",
54
58
  label: "StyledLabel"
55
- })("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, useMemo } from 'react';\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 { IFolder } from '@ndla/types-learningpath-api';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType } from './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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  editable?: boolean;\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  editable,\n  menuItems,\n  folders,\n  framed,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    setNewFolderParentId(undefined);\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId(undefined);\n  };\n\n  const canAddFolder =\n    editable && selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <div>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureStyledWrapper aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          editable={editable}\n          focusedFolderId={focusedId}\n          menuItems={menuItems}\n          folders={folders}\n          level={0}\n          loading={loading}\n          selectedFolder={selectedFolder}\n          maximumLevelsOfFoldersAllowed={maximumLevelsOfFoldersAllowed}\n          newFolderParentId={newFolderParentId}\n          onCancelNewFolder={onCancelNewFolder}\n          onCloseFolder={onCloseFolder}\n          onOpenFolder={onOpenFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          onSelectFolder={onSelectFolder}\n          openFolders={openFolders}\n          openOnFolderClick={openOnFolderClick}\n          setFocusedId={setFocusedId}\n          setSelectedFolder={setSelectedFolder}\n          targetResource={targetResource}\n          visibleFolders={visibleFolderIds}\n          framed={framed}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={\n              canAddFolder\n                ? t('myNdla.newFolderUnder', {\n                    folderName: selectedFolder?.name,\n                  })\n                : t('treeStructure.maxFoldersAlreadyAdded')\n            }>\n            <AddButton\n              disabled={!canAddFolder}\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n              {t('myNdla.newFolder')}\n            </AddButton>\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
56
- var AddFolderWrapper = (0, _styledBase["default"])("div", {
59
+ })("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":"AAyBgC","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
60
+ var StyledRow = (0, _styledBase["default"])("div", {
57
61
  target: "e1dg1gdn1",
58
- label: "AddFolderWrapper"
59
- })("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, useMemo } from 'react';\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 { IFolder } from '@ndla/types-learningpath-api';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType } from './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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  editable?: boolean;\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  editable,\n  menuItems,\n  folders,\n  framed,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    setNewFolderParentId(undefined);\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId(undefined);\n  };\n\n  const canAddFolder =\n    editable && selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <div>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureStyledWrapper aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          editable={editable}\n          focusedFolderId={focusedId}\n          menuItems={menuItems}\n          folders={folders}\n          level={0}\n          loading={loading}\n          selectedFolder={selectedFolder}\n          maximumLevelsOfFoldersAllowed={maximumLevelsOfFoldersAllowed}\n          newFolderParentId={newFolderParentId}\n          onCancelNewFolder={onCancelNewFolder}\n          onCloseFolder={onCloseFolder}\n          onOpenFolder={onOpenFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          onSelectFolder={onSelectFolder}\n          openFolders={openFolders}\n          openOnFolderClick={openOnFolderClick}\n          setFocusedId={setFocusedId}\n          setSelectedFolder={setSelectedFolder}\n          targetResource={targetResource}\n          visibleFolders={visibleFolderIds}\n          framed={framed}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={\n              canAddFolder\n                ? t('myNdla.newFolderUnder', {\n                    folderName: selectedFolder?.name,\n                  })\n                : t('treeStructure.maxFoldersAlreadyAdded')\n            }>\n            <AddButton\n              disabled={!canAddFolder}\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n              {t('myNdla.newFolder')}\n            </AddButton>\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
60
-
61
- var TreeStructure = function TreeStructure(_ref) {
62
- var defaultOpenFolders = _ref.defaultOpenFolders,
63
- editable = _ref.editable,
64
- menuItems = _ref.menuItems,
65
- folders = _ref.folders,
66
- framed = _ref.framed,
67
- label = _ref.label,
68
- loading = _ref.loading,
69
- _ref$maximumLevelsOfF = _ref.maximumLevelsOfFoldersAllowed,
70
- maximumLevelsOfFoldersAllowed = _ref$maximumLevelsOfF === void 0 ? MAX_LEVEL_FOR_FOLDERS : _ref$maximumLevelsOfF,
71
- onNewFolder = _ref.onNewFolder,
72
- onSelectFolder = _ref.onSelectFolder,
73
- openOnFolderClick = _ref.openOnFolderClick,
74
- targetResource = _ref.targetResource;
62
+ label: "StyledRow"
63
+ })("display:flex;justify-content:space-between;padding:", _core.spacing.xxsmall, ";border-bottom:", function (_ref) {
64
+ var isOpen = _ref.isOpen;
65
+ return isOpen && "1px solid ".concat(_core.colors.brand.tertiary);
66
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAiC4C","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
67
+ var StyledTreeStructure = (0, _styledBase["default"])("div", {
68
+ target: "e1dg1gdn2",
69
+ label: "StyledTreeStructure"
70
+ })(process.env.NODE_ENV === "production" ? {
71
+ name: "k0sogd",
72
+ styles: "flex:1;display:flex;flex-direction:column;"
73
+ } : {
74
+ name: "k0sogd",
75
+ styles: "flex:1;display:flex;flex-direction:column;",
76
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAwCsC","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */",
77
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
78
+ });
79
+ var TreeStructureWrapper = (0, _styledBase["default"])("div", {
80
+ target: "e1dg1gdn3",
81
+ label: "TreeStructureWrapper"
82
+ })("display:flex;flex-direction:column;", function (_ref2) {
83
+ var type = _ref2.type;
84
+ return (type === 'normal' || type === 'picker') && /*#__PURE__*/(0, _core2.css)("overflow:hidden;border:1px solid ", _core.colors.brand.neutral7, ";border-radius:", _core.misc.borderRadius, ";scroll-behavior:smooth;;label:TreeStructureWrapper;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAmDO","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
85
+ }, " transition:", _core.misc.transition["default"], ";&:focus-within{border-color:", _core.colors.brand.tertiary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA8CoE","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
86
+ var ScrollableDiv = (0, _styledBase["default"])("div", {
87
+ target: "e1dg1gdn4",
88
+ label: "ScrollableDiv"
89
+ })(function (_ref3) {
90
+ var type = _ref3.type;
91
+ return (type === 'picker' || type === 'normal') && /*#__PURE__*/(0, _core2.css)("overflow:overlay;::-webkit-scrollbar{width:", _core.spacing.small, ";}::-webkit-scrollbar-thumb{border:4px solid transparent;border-radius:14px;background-clip:padding-box;padding:0 4px;background-color:", _core.colors.brand.neutral7, ";};label:ScrollableDiv;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAoEO","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
92
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAiEoD","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */");
93
+ var StyledSelectedFolder = ( /*#__PURE__*/0, _styledBase["default"])(_button.ButtonV2, {
94
+ target: "e1dg1gdn5",
95
+ label: "StyledSelectedFolder"
96
+ })(process.env.NODE_ENV === "production" ? {
97
+ name: "wj4dip",
98
+ styles: "flex:1;justify-content:flex-start;:hover,:focus{background:none;box-shadow:none;border-color:transparent;}"
99
+ } : {
100
+ name: "wj4dip",
101
+ styles: "flex:1;justify-content:flex-start;:hover,:focus{background:none;box-shadow:none;border-color:transparent;}",
102
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAmF2C","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */",
103
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
104
+ });
105
+ var StyledAddFolderButton = ( /*#__PURE__*/0, _styledBase["default"])(_button.ButtonV2, {
106
+ target: "e1dg1gdn6",
107
+ label: "StyledAddFolderButton"
108
+ })(process.env.NODE_ENV === "production" ? {
109
+ name: "it7ogd",
110
+ styles: "&,&:disabled{border-color:transparent;}"
111
+ } : {
112
+ name: "it7ogd",
113
+ styles: "&,&:disabled{border-color:transparent;}",
114
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA8F4C","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */",
115
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
116
+ });
117
+ var StyledPlus = ( /*#__PURE__*/0, _styledBase["default"])(_action.Plus, {
118
+ target: "e1dg1gdn7",
119
+ label: "StyledPlus"
120
+ })(process.env.NODE_ENV === "production" ? {
121
+ name: "1m01c8l",
122
+ styles: "height:24px;width:24px;"
123
+ } : {
124
+ name: "1m01c8l",
125
+ styles: "height:24px;width:24px;",
126
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAqG+B","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, useMemo, useRef } from 'react';\nimport { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';\nimport { Plus } from '@ndla/icons/action';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { colors, fonts, misc, spacing } from '@ndla/core';\nimport { css } from '@emotion/core';\nimport { uniq } from 'lodash';\nimport { IFolder } from '@ndla/types-learningpath-api';\nimport FolderItems from './FolderItems';\nimport { flattenFolders } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\ninterface StyledRowProps {\n  isOpen: boolean;\n}\n\nconst StyledRow = styled.div<StyledRowProps>`\n  display: flex;\n  justify-content: space-between;\n  padding: ${spacing.xxsmall};\n  border-bottom: ${({ isOpen }) => isOpen && `1px solid ${colors.brand.tertiary}`};\n`;\n\nconst StyledTreeStructure = styled.div`\n  flex: 1;\n  display: flex;\n  flex-direction: column;\n`;\n\nconst TreeStructureWrapper = styled.div<{ type: TreeStructureType }>`\n  display: flex;\n  flex-direction: column;\n  ${({ type }) =>\n    (type === 'normal' || type === 'picker') &&\n    css`\n      overflow: hidden;\n      border: 1px solid ${colors.brand.neutral7};\n      border-radius: ${misc.borderRadius};\n      scroll-behavior: smooth;\n    `}\n  transition: ${misc.transition.default};\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    (type === 'picker' || type === 'normal') &&\n    css`\n      overflow: overlay;\n      ::-webkit-scrollbar {\n        width: ${spacing.small};\n      }\n      ::-webkit-scrollbar-thumb {\n        border: 4px solid transparent;\n        border-radius: 14px;\n        background-clip: padding-box;\n        padding: 0 4px;\n        background-color: ${colors.brand.neutral7};\n      }\n    `}\n`;\n\nconst StyledSelectedFolder = styled(Button)`\n  flex: 1;\n  justify-content: flex-start;\n  :hover,\n  :focus {\n    background: none;\n    box-shadow: none;\n    border-color: transparent;\n  }\n`;\n\nconst StyledAddFolderButton = styled(Button)`\n  &,\n  &:disabled {\n    border-color: transparent;\n  }\n`;\n\nconst StyledPlus = styled(Plus)`\n  height: 24px;\n  width: 24px;\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maximumLevelsOfFoldersAllowed?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maximumLevelsOfFoldersAllowed = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  openOnFolderClick,\n  targetResource,\n  type = 'normal',\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n\n  const ref = useRef<HTMLDivElement>(null);\n\n  const defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];\n\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n\n  const [newFolderParentId, setNewFolderParentId] = useState<string | undefined>();\n  const [focusedId, setFocusedId] = useState<string | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type !== 'picker');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\n  const visibleFolderIds = flattenedFolders.map((folder) => folder.id);\n\n  useEffect(() => {\n    const handleClickOutside = (e: MouseEvent) => {\n      if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {\n        setShowTree(false);\n      }\n    };\n    if (type === 'picker') {\n      document.addEventListener('mousedown', handleClickOutside);\n      return () => {\n        document.removeEventListener('mousedown', handleClickOutside);\n      };\n    }\n  }, [ref, type]);\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      if (!defaultOpenFolders.every((element) => openFolders.includes(element))) {\n        setOpenFolders((prev) => {\n          return uniq(defaultOpenFolders.concat(prev));\n        });\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    setNewFolderParentId(undefined);\n  }, [selectedFolder]);\n\n  useEffect(() => {\n    if (defaultSelectedFolderId !== undefined) {\n      const selected = flattenFolders(folders).find((folder) => folder.id === defaultSelectedFolderId);\n      if (selected) {\n        setSelectedFolder(selected);\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolderParentId(undefined);\n    }\n  }, [loading]);\n\n  const onCloseFolder = (id: string) => {\n    const closedFolder = flattenedFolders.find((folder) => folder.id === id);\n\n    if (closedFolder) {\n      const subFolders = closedFolder.subfolders && flattenFolders(closedFolder.subfolders);\n      if (subFolders.some((folder) => folder.id === selectedFolder?.id)) {\n        if (onSelectFolder) {\n          setSelectedFolder(closedFolder);\n          onSelectFolder(closedFolder.id);\n        }\n        setFocusedId(closedFolder.id);\n      }\n    }\n    setOpenFolders(openFolders.filter((folderId) => folderId !== id));\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\n  };\n\n  const onSaveNewFolder = (name: string, parentId: string) => {\n    onNewFolder?.(name, parentId).then((newFolder) => {\n      if (newFolder) {\n        setNewFolderParentId?.(undefined);\n        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedId(newFolder.id);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <StyledTreeStructure ref={ref}>\n      {label && <StyledLabel>{label}</StyledLabel>}\n      <TreeStructureWrapper aria-label={label} type={type}>\n        {type === 'picker' && (\n          <StyledRow isOpen={showTree}>\n            <StyledSelectedFolder\n              variant=\"ghost\"\n              colorTheme=\"light\"\n              fontWeight=\"normal\"\n              shape=\"sharp\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}>\n              {selectedFolder?.name}\n            </StyledSelectedFolder>\n            {onNewFolder && showTree && (\n              <Tooltip\n                tooltip={\n                  canAddFolder\n                    ? t('myNdla.newFolderUnder', {\n                        folderName: selectedFolder?.name,\n                      })\n                    : t('treeStructure.maxFoldersAlreadyAdded')\n                }>\n                <StyledAddFolderButton\n                  variant=\"outline\"\n                  shape=\"pill\"\n                  disabled={!canAddFolder}\n                  aria-label={\n                    canAddFolder\n                      ? t('myNdla.newFolderUnder', {\n                          folderName: selectedFolder?.name,\n                        })\n                      : t('treeStructure.maxFoldersAlreadyAdded')\n                  }\n                  onClick={() => setNewFolderParentId(selectedFolder?.id)}>\n                  <StyledPlus /> {t('myNdla.newFolder')}\n                </StyledAddFolderButton>\n              </Tooltip>\n            )}\n            <IconButtonDualStates\n              ariaLabelActive={t('treeStructure.hideFolders')}\n              ariaLabelInActive={t('treeStructure.showFolders')}\n              active={showTree}\n              variant=\"ghost\"\n              colorTheme=\"greyLighter\"\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              onClick={() => {\n                setShowTree(!showTree);\n              }}\n            />\n          </StyledRow>\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolderId={focusedId}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maximumLevelsOfFoldersAllowed}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              openOnFolderClick={openOnFolderClick}\n              setFocusedId={setFocusedId}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={visibleFolderIds}\n              type={type}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */",
127
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
128
+ });
129
+
130
+ var TreeStructure = function TreeStructure(_ref4) {
131
+ var defaultOpenFolders = _ref4.defaultOpenFolders,
132
+ folders = _ref4.folders,
133
+ label = _ref4.label,
134
+ loading = _ref4.loading,
135
+ _ref4$maximumLevelsOf = _ref4.maximumLevelsOfFoldersAllowed,
136
+ maximumLevelsOfFoldersAllowed = _ref4$maximumLevelsOf === void 0 ? MAX_LEVEL_FOR_FOLDERS : _ref4$maximumLevelsOf,
137
+ onNewFolder = _ref4.onNewFolder,
138
+ onSelectFolder = _ref4.onSelectFolder,
139
+ openOnFolderClick = _ref4.openOnFolderClick,
140
+ targetResource = _ref4.targetResource,
141
+ _ref4$type = _ref4.type,
142
+ type = _ref4$type === void 0 ? 'normal' : _ref4$type;
75
143
 
76
144
  var _useTranslation = (0, _reactI18next.useTranslation)(),
77
145
  t = _useTranslation.t;
78
146
 
147
+ var ref = (0, _react.useRef)(null);
79
148
  var defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];
80
149
 
81
150
  var _useState = (0, _react.useState)(defaultOpenFolders || []),
@@ -98,12 +167,31 @@ var TreeStructure = function TreeStructure(_ref) {
98
167
  selectedFolder = _useState8[0],
99
168
  setSelectedFolder = _useState8[1];
100
169
 
170
+ var _useState9 = (0, _react.useState)(type !== 'picker'),
171
+ _useState10 = _slicedToArray(_useState9, 2),
172
+ showTree = _useState10[0],
173
+ setShowTree = _useState10[1];
174
+
101
175
  var flattenedFolders = (0, _react.useMemo)(function () {
102
176
  return (0, _helperFunctions.flattenFolders)(folders, openFolders);
103
177
  }, [folders, openFolders]);
104
178
  var visibleFolderIds = flattenedFolders.map(function (folder) {
105
179
  return folder.id;
106
180
  });
181
+ (0, _react.useEffect)(function () {
182
+ var handleClickOutside = function handleClickOutside(e) {
183
+ if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {
184
+ setShowTree(false);
185
+ }
186
+ };
187
+
188
+ if (type === 'picker') {
189
+ document.addEventListener('mousedown', handleClickOutside);
190
+ return function () {
191
+ document.removeEventListener('mousedown', handleClickOutside);
192
+ };
193
+ }
194
+ }, [ref, type]);
107
195
  (0, _react.useEffect)(function () {
108
196
  if (defaultOpenFolders) {
109
197
  if (!defaultOpenFolders.every(function (element) {
@@ -116,6 +204,9 @@ var TreeStructure = function TreeStructure(_ref) {
116
204
  } // eslint-disable-next-line react-hooks/exhaustive-deps
117
205
 
118
206
  }, [defaultOpenFolders]);
207
+ (0, _react.useEffect)(function () {
208
+ setNewFolderParentId(undefined);
209
+ }, [selectedFolder]);
119
210
  (0, _react.useEffect)(function () {
120
211
  if (defaultSelectedFolderId !== undefined) {
121
212
  var selected = (0, _helperFunctions.flattenFolders)(folders).find(function (folder) {
@@ -164,9 +255,9 @@ var TreeStructure = function TreeStructure(_ref) {
164
255
  };
165
256
 
166
257
  var onSaveNewFolder = function onSaveNewFolder(name, parentId) {
167
- setNewFolderParentId(undefined);
168
258
  onNewFolder === null || onNewFolder === void 0 ? void 0 : onNewFolder(name, parentId).then(function (newFolder) {
169
259
  if (newFolder) {
260
+ setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
170
261
  setSelectedFolder(newFolder);
171
262
  onSelectFolder === null || onSelectFolder === void 0 ? void 0 : onSelectFolder(newFolder.id);
172
263
  setFocusedId(newFolder.id);
@@ -176,23 +267,60 @@ var TreeStructure = function TreeStructure(_ref) {
176
267
  };
177
268
 
178
269
  var onCancelNewFolder = function onCancelNewFolder() {
179
- setNewFolderParentId(undefined);
270
+ setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
180
271
  };
181
272
 
182
- var canAddFolder = editable && selectedFolder && (selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.breadcrumbs.length) < (maximumLevelsOfFoldersAllowed || 1);
183
- return (0, _core2.jsx)("div", null, label && (0, _core2.jsx)(StyledLabel, null, label), (0, _core2.jsx)(_TreeStructureWrapper["default"], {
184
- "aria-label": "Menu tree",
185
- role: "tree",
186
- framed: framed
273
+ var canAddFolder = selectedFolder && (selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.breadcrumbs.length) < (maximumLevelsOfFoldersAllowed || 1);
274
+ return (0, _core2.jsx)(StyledTreeStructure, {
275
+ ref: ref
276
+ }, label && (0, _core2.jsx)(StyledLabel, null, label), (0, _core2.jsx)(TreeStructureWrapper, {
277
+ "aria-label": label,
278
+ type: type
279
+ }, type === 'picker' && (0, _core2.jsx)(StyledRow, {
280
+ isOpen: showTree
281
+ }, (0, _core2.jsx)(StyledSelectedFolder, {
282
+ variant: "ghost",
283
+ colorTheme: "light",
284
+ fontWeight: "normal",
285
+ shape: "sharp",
286
+ onClick: function onClick() {
287
+ setShowTree(!showTree);
288
+ }
289
+ }, selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.name), onNewFolder && showTree && (0, _core2.jsx)(_tooltip["default"], {
290
+ tooltip: canAddFolder ? t('myNdla.newFolderUnder', {
291
+ folderName: selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.name
292
+ }) : t('treeStructure.maxFoldersAlreadyAdded')
293
+ }, (0, _core2.jsx)(StyledAddFolderButton, {
294
+ variant: "outline",
295
+ shape: "pill",
296
+ disabled: !canAddFolder,
297
+ "aria-label": canAddFolder ? t('myNdla.newFolderUnder', {
298
+ folderName: selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.name
299
+ }) : t('treeStructure.maxFoldersAlreadyAdded'),
300
+ onClick: function onClick() {
301
+ return setNewFolderParentId(selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.id);
302
+ }
303
+ }, (0, _core2.jsx)(StyledPlus, null), " ", t('myNdla.newFolder'))), (0, _core2.jsx)(_button.IconButtonDualStates, {
304
+ ariaLabelActive: t('treeStructure.hideFolders'),
305
+ ariaLabelInActive: t('treeStructure.showFolders'),
306
+ active: showTree,
307
+ variant: "ghost",
308
+ colorTheme: "greyLighter",
309
+ inactiveIcon: (0, _core2.jsx)(_common.ChevronDown, null),
310
+ activeIcon: (0, _core2.jsx)(_common.ChevronUp, null),
311
+ size: "small",
312
+ onClick: function onClick() {
313
+ setShowTree(!showTree);
314
+ }
315
+ })), showTree && (0, _core2.jsx)(ScrollableDiv, {
316
+ type: type
187
317
  }, (0, _core2.jsx)(_FolderItems["default"], {
188
- editable: editable,
189
318
  focusedFolderId: focusedId,
190
- menuItems: menuItems,
191
319
  folders: folders,
192
320
  level: 0,
193
321
  loading: loading,
194
322
  selectedFolder: selectedFolder,
195
- maximumLevelsOfFoldersAllowed: maximumLevelsOfFoldersAllowed,
323
+ maxLevel: maximumLevelsOfFoldersAllowed,
196
324
  newFolderParentId: newFolderParentId,
197
325
  onCancelNewFolder: onCancelNewFolder,
198
326
  onCloseFolder: onCloseFolder,
@@ -205,18 +333,8 @@ var TreeStructure = function TreeStructure(_ref) {
205
333
  setSelectedFolder: setSelectedFolder,
206
334
  targetResource: targetResource,
207
335
  visibleFolders: visibleFolderIds,
208
- framed: framed
209
- })), editable && (0, _core2.jsx)(AddFolderWrapper, null, (0, _core2.jsx)(_tooltip["default"], {
210
- tooltip: canAddFolder ? t('myNdla.newFolderUnder', {
211
- folderName: selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.name
212
- }) : t('treeStructure.maxFoldersAlreadyAdded')
213
- }, (0, _core2.jsx)(_button.AddButton, {
214
- disabled: !canAddFolder,
215
- "aria-label": t('myNdla.newFolder'),
216
- onClick: function onClick() {
217
- return setNewFolderParentId(selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.id);
218
- }
219
- }, t('myNdla.newFolder')))));
336
+ type: type
337
+ }))));
220
338
  };
221
339
 
222
340
  var _default = TreeStructure;