@ndla/ui 26.0.0 → 27.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/Breadcrumb/Breadcrumb.js +3 -4
  2. package/es/MyNdla/Resource/Folder.js +32 -14
  3. package/es/Resource/BlockResource.js +43 -61
  4. package/es/Resource/ListResource.js +44 -23
  5. package/es/Resource/resourceComponents.js +64 -38
  6. package/es/TreeStructure/ComboboxButton.js +162 -0
  7. package/es/TreeStructure/FolderItem.js +98 -78
  8. package/es/TreeStructure/FolderItems.js +25 -14
  9. package/es/TreeStructure/FolderNameInput.js +40 -33
  10. package/es/TreeStructure/NavigationLink.js +18 -10
  11. package/es/TreeStructure/TreeStructure.js +92 -165
  12. package/es/TreeStructure/arrowNavigation.js +3 -3
  13. package/es/TreeStructure/helperFunctions.js +3 -0
  14. package/es/locale/messages-en.js +8 -2
  15. package/es/locale/messages-nb.js +8 -2
  16. package/es/locale/messages-nn.js +8 -2
  17. package/es/locale/messages-se.js +8 -2
  18. package/es/locale/messages-sma.js +8 -2
  19. package/lib/Breadcrumb/Breadcrumb.js +3 -5
  20. package/lib/MyNdla/Resource/Folder.d.ts +2 -1
  21. package/lib/MyNdla/Resource/Folder.js +37 -14
  22. package/lib/Resource/BlockResource.d.ts +2 -1
  23. package/lib/Resource/BlockResource.js +48 -61
  24. package/lib/Resource/ListResource.d.ts +2 -1
  25. package/lib/Resource/ListResource.js +49 -23
  26. package/lib/Resource/resourceComponents.d.ts +6 -1
  27. package/lib/Resource/resourceComponents.js +64 -37
  28. package/lib/TreeStructure/ComboboxButton.d.ts +28 -0
  29. package/lib/TreeStructure/ComboboxButton.js +176 -0
  30. package/lib/TreeStructure/FolderItem.d.ts +1 -1
  31. package/lib/TreeStructure/FolderItem.js +99 -77
  32. package/lib/TreeStructure/FolderItems.d.ts +4 -2
  33. package/lib/TreeStructure/FolderItems.js +26 -14
  34. package/lib/TreeStructure/FolderNameInput.d.ts +3 -1
  35. package/lib/TreeStructure/FolderNameInput.js +41 -32
  36. package/lib/TreeStructure/NavigationLink.d.ts +1 -1
  37. package/lib/TreeStructure/NavigationLink.js +18 -10
  38. package/lib/TreeStructure/TreeStructure.d.ts +2 -2
  39. package/lib/TreeStructure/TreeStructure.js +92 -165
  40. package/lib/TreeStructure/arrowNavigation.d.ts +2 -1
  41. package/lib/TreeStructure/arrowNavigation.js +3 -3
  42. package/lib/TreeStructure/helperFunctions.d.ts +2 -1
  43. package/lib/TreeStructure/helperFunctions.js +8 -2
  44. package/lib/TreeStructure/types.d.ts +6 -7
  45. package/lib/locale/messages-en.d.ts +6 -0
  46. package/lib/locale/messages-en.js +8 -2
  47. package/lib/locale/messages-nb.d.ts +6 -0
  48. package/lib/locale/messages-nb.js +8 -2
  49. package/lib/locale/messages-nn.d.ts +6 -0
  50. package/lib/locale/messages-nn.js +8 -2
  51. package/lib/locale/messages-se.d.ts +6 -0
  52. package/lib/locale/messages-se.js +8 -2
  53. package/lib/locale/messages-sma.d.ts +6 -0
  54. package/lib/locale/messages-sma.js +8 -2
  55. package/package.json +11 -11
  56. package/src/Breadcrumb/Breadcrumb.tsx +1 -2
  57. package/src/MyNdla/Resource/Folder.tsx +21 -5
  58. package/src/Resource/BlockResource.tsx +43 -33
  59. package/src/Resource/ListResource.tsx +37 -29
  60. package/src/Resource/resourceComponents.tsx +60 -26
  61. package/src/TreeStructure/ComboboxButton.tsx +189 -0
  62. package/src/TreeStructure/FolderItem.tsx +89 -70
  63. package/src/TreeStructure/FolderItems.tsx +36 -16
  64. package/src/TreeStructure/FolderNameInput.tsx +43 -18
  65. package/src/TreeStructure/NavigationLink.tsx +17 -10
  66. package/src/TreeStructure/TreeStructure.tsx +63 -139
  67. package/src/TreeStructure/arrowNavigation.ts +7 -6
  68. package/src/TreeStructure/helperFunctions.ts +5 -1
  69. package/src/TreeStructure/types.ts +6 -7
  70. package/src/locale/messages-en.ts +7 -0
  71. package/src/locale/messages-nb.ts +6 -0
  72. package/src/locale/messages-nn.ts +6 -0
  73. package/src/locale/messages-se.ts +7 -0
  74. package/src/locale/messages-sma.ts +7 -0
@@ -11,16 +11,6 @@ var _styledBase = _interopRequireDefault(require("@emotion/styled-base"));
11
11
 
12
12
  var _react = _interopRequireWildcard(require("react"));
13
13
 
14
- var _button = require("@ndla/button");
15
-
16
- var _action = require("@ndla/icons/action");
17
-
18
- var _common = require("@ndla/icons/common");
19
-
20
- var _tooltip = _interopRequireDefault(require("@ndla/tooltip"));
21
-
22
- var _reactI18next = require("react-i18next");
23
-
24
14
  var _core = require("@ndla/core");
25
15
 
26
16
  var _core2 = require("@emotion/core");
@@ -31,6 +21,8 @@ var _FolderItems = _interopRequireDefault(require("./FolderItems"));
31
21
 
32
22
  var _helperFunctions = require("./helperFunctions");
33
23
 
24
+ var _ComboboxButton = _interopRequireDefault(require("./ComboboxButton"));
25
+
34
26
  function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; }
35
27
 
36
28
  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; }
@@ -56,16 +48,9 @@ exports.MAX_LEVEL_FOR_FOLDERS = MAX_LEVEL_FOR_FOLDERS;
56
48
  var StyledLabel = (0, _styledBase["default"])("label", {
57
49
  target: "e1dg1gdn0",
58
50
  label: "StyledLabel"
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", {
61
- target: "e1dg1gdn1",
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"]} */"));
51
+ })("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":"AAqBgC","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 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, treestructureId } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\nimport ComboboxButton from './ComboboxButton';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\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 === '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`;\n\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\n\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    type === 'picker' &&\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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maxLevel?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  targetResource,\n  type,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(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 [focusedFolder, setFocusedFolder] = useState<FolderType | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type === 'navigation');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\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        if (type === 'picker') {\n          setFocusedFolder(selected);\n        }\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  const onToggleTree = (open: boolean) => {\n    setShowTree(open);\n    if (!open) {\n      setNewFolderParentId(undefined);\n    }\n  };\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        setFocusedFolder(closedFolder);\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        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedFolder(newFolder);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n        setNewFolderParentId?.(undefined);\n        ref.current?.focus();\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus();\n  };\n\n  const setFolderFocus = (folder: FolderType, focus?: boolean) => {\n    setFocusedFolder(folder);\n\n    if (focus) {\n      ref.current?.focus();\n    }\n  };\n\n  return (\n    <StyledTreeStructure>\n      {label && <StyledLabel id={treestructureId(type, 'label')}>{label}</StyledLabel>}\n      <TreeStructureWrapper\n        aria-label={label}\n        type={type}\n        onBlur={(e) => {\n          if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {\n            onToggleTree(false);\n          }\n        }}>\n        {type === 'picker' && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            focusedFolder={focusedFolder}\n            selectedFolder={selectedFolder}\n            setSelectedFolder={setSelectedFolder}\n            setFocusedFolder={setFocusedFolder}\n            onToggleTree={onToggleTree}\n            flattenedFolders={flattenedFolders}\n            onCloseFolder={onCloseFolder}\n            onOpenFolder={onOpenFolder}\n            onNewFolder={onNewFolder}\n            maxLevel={maxLevel}\n            setNewFolderParentId={setNewFolderParentId}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolder={focusedFolder}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maxLevel}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              setFocusedFolder={setFolderFocus}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
67
52
  var StyledTreeStructure = (0, _styledBase["default"])("div", {
68
- target: "e1dg1gdn2",
53
+ target: "e1dg1gdn1",
69
54
  label: "StyledTreeStructure"
70
55
  })(process.env.NODE_ENV === "production" ? {
71
56
  name: "k0sogd",
@@ -73,77 +58,35 @@ var StyledTreeStructure = (0, _styledBase["default"])("div", {
73
58
  } : {
74
59
  name: "k0sogd",
75
60
  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"]} */",
61
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAyBsC","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 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, treestructureId } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\nimport ComboboxButton from './ComboboxButton';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\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 === '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`;\n\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\n\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    type === 'picker' &&\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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maxLevel?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  targetResource,\n  type,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(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 [focusedFolder, setFocusedFolder] = useState<FolderType | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type === 'navigation');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\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        if (type === 'picker') {\n          setFocusedFolder(selected);\n        }\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  const onToggleTree = (open: boolean) => {\n    setShowTree(open);\n    if (!open) {\n      setNewFolderParentId(undefined);\n    }\n  };\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        setFocusedFolder(closedFolder);\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        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedFolder(newFolder);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n        setNewFolderParentId?.(undefined);\n        ref.current?.focus();\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus();\n  };\n\n  const setFolderFocus = (folder: FolderType, focus?: boolean) => {\n    setFocusedFolder(folder);\n\n    if (focus) {\n      ref.current?.focus();\n    }\n  };\n\n  return (\n    <StyledTreeStructure>\n      {label && <StyledLabel id={treestructureId(type, 'label')}>{label}</StyledLabel>}\n      <TreeStructureWrapper\n        aria-label={label}\n        type={type}\n        onBlur={(e) => {\n          if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {\n            onToggleTree(false);\n          }\n        }}>\n        {type === 'picker' && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            focusedFolder={focusedFolder}\n            selectedFolder={selectedFolder}\n            setSelectedFolder={setSelectedFolder}\n            setFocusedFolder={setFocusedFolder}\n            onToggleTree={onToggleTree}\n            flattenedFolders={flattenedFolders}\n            onCloseFolder={onCloseFolder}\n            onOpenFolder={onOpenFolder}\n            onNewFolder={onNewFolder}\n            maxLevel={maxLevel}\n            setNewFolderParentId={setNewFolderParentId}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolder={focusedFolder}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maxLevel}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              setFocusedFolder={setFolderFocus}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */",
77
62
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
78
63
  });
79
64
  var TreeStructureWrapper = (0, _styledBase["default"])("div", {
80
- target: "e1dg1gdn3",
65
+ target: "e1dg1gdn2",
81
66
  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"]} */"));
67
+ })("display:flex;flex-direction:column;", function (_ref) {
68
+ var type = _ref.type;
69
+ return 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":"AAoCO","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 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, treestructureId } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\nimport ComboboxButton from './ComboboxButton';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\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 === '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`;\n\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\n\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    type === 'picker' &&\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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maxLevel?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  targetResource,\n  type,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(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 [focusedFolder, setFocusedFolder] = useState<FolderType | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type === 'navigation');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\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        if (type === 'picker') {\n          setFocusedFolder(selected);\n        }\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  const onToggleTree = (open: boolean) => {\n    setShowTree(open);\n    if (!open) {\n      setNewFolderParentId(undefined);\n    }\n  };\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        setFocusedFolder(closedFolder);\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        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedFolder(newFolder);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n        setNewFolderParentId?.(undefined);\n        ref.current?.focus();\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus();\n  };\n\n  const setFolderFocus = (folder: FolderType, focus?: boolean) => {\n    setFocusedFolder(folder);\n\n    if (focus) {\n      ref.current?.focus();\n    }\n  };\n\n  return (\n    <StyledTreeStructure>\n      {label && <StyledLabel id={treestructureId(type, 'label')}>{label}</StyledLabel>}\n      <TreeStructureWrapper\n        aria-label={label}\n        type={type}\n        onBlur={(e) => {\n          if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {\n            onToggleTree(false);\n          }\n        }}>\n        {type === 'picker' && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            focusedFolder={focusedFolder}\n            selectedFolder={selectedFolder}\n            setSelectedFolder={setSelectedFolder}\n            setFocusedFolder={setFocusedFolder}\n            onToggleTree={onToggleTree}\n            flattenedFolders={flattenedFolders}\n            onCloseFolder={onCloseFolder}\n            onOpenFolder={onOpenFolder}\n            onNewFolder={onNewFolder}\n            maxLevel={maxLevel}\n            setNewFolderParentId={setNewFolderParentId}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolder={focusedFolder}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maxLevel}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              setFocusedFolder={setFolderFocus}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
70
+ }, " 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":"AA+BoE","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 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, treestructureId } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\nimport ComboboxButton from './ComboboxButton';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\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 === '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`;\n\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\n\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    type === 'picker' &&\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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maxLevel?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  targetResource,\n  type,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(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 [focusedFolder, setFocusedFolder] = useState<FolderType | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type === 'navigation');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\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        if (type === 'picker') {\n          setFocusedFolder(selected);\n        }\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  const onToggleTree = (open: boolean) => {\n    setShowTree(open);\n    if (!open) {\n      setNewFolderParentId(undefined);\n    }\n  };\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        setFocusedFolder(closedFolder);\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        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedFolder(newFolder);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n        setNewFolderParentId?.(undefined);\n        ref.current?.focus();\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus();\n  };\n\n  const setFolderFocus = (folder: FolderType, focus?: boolean) => {\n    setFocusedFolder(folder);\n\n    if (focus) {\n      ref.current?.focus();\n    }\n  };\n\n  return (\n    <StyledTreeStructure>\n      {label && <StyledLabel id={treestructureId(type, 'label')}>{label}</StyledLabel>}\n      <TreeStructureWrapper\n        aria-label={label}\n        type={type}\n        onBlur={(e) => {\n          if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {\n            onToggleTree(false);\n          }\n        }}>\n        {type === 'picker' && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            focusedFolder={focusedFolder}\n            selectedFolder={selectedFolder}\n            setSelectedFolder={setSelectedFolder}\n            setFocusedFolder={setFocusedFolder}\n            onToggleTree={onToggleTree}\n            flattenedFolders={flattenedFolders}\n            onCloseFolder={onCloseFolder}\n            onOpenFolder={onOpenFolder}\n            onNewFolder={onNewFolder}\n            maxLevel={maxLevel}\n            setNewFolderParentId={setNewFolderParentId}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolder={focusedFolder}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maxLevel}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              setFocusedFolder={setFolderFocus}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
86
71
  var ScrollableDiv = (0, _styledBase["default"])("div", {
87
- target: "e1dg1gdn4",
72
+ target: "e1dg1gdn3",
88
73
  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;
143
-
144
- var _useTranslation = (0, _reactI18next.useTranslation)(),
145
- t = _useTranslation.t;
146
-
74
+ })(function (_ref2) {
75
+ var type = _ref2.type;
76
+ return type === 'picker' && /*#__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":"AAuDO","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 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, treestructureId } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\nimport ComboboxButton from './ComboboxButton';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\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 === '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`;\n\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\n\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    type === 'picker' &&\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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maxLevel?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  targetResource,\n  type,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(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 [focusedFolder, setFocusedFolder] = useState<FolderType | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type === 'navigation');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\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        if (type === 'picker') {\n          setFocusedFolder(selected);\n        }\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  const onToggleTree = (open: boolean) => {\n    setShowTree(open);\n    if (!open) {\n      setNewFolderParentId(undefined);\n    }\n  };\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        setFocusedFolder(closedFolder);\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        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedFolder(newFolder);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n        setNewFolderParentId?.(undefined);\n        ref.current?.focus();\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus();\n  };\n\n  const setFolderFocus = (folder: FolderType, focus?: boolean) => {\n    setFocusedFolder(folder);\n\n    if (focus) {\n      ref.current?.focus();\n    }\n  };\n\n  return (\n    <StyledTreeStructure>\n      {label && <StyledLabel id={treestructureId(type, 'label')}>{label}</StyledLabel>}\n      <TreeStructureWrapper\n        aria-label={label}\n        type={type}\n        onBlur={(e) => {\n          if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {\n            onToggleTree(false);\n          }\n        }}>\n        {type === 'picker' && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            focusedFolder={focusedFolder}\n            selectedFolder={selectedFolder}\n            setSelectedFolder={setSelectedFolder}\n            setFocusedFolder={setFocusedFolder}\n            onToggleTree={onToggleTree}\n            flattenedFolders={flattenedFolders}\n            onCloseFolder={onCloseFolder}\n            onOpenFolder={onOpenFolder}\n            onNewFolder={onNewFolder}\n            maxLevel={maxLevel}\n            setNewFolderParentId={setNewFolderParentId}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolder={focusedFolder}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maxLevel}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              setFocusedFolder={setFolderFocus}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
77
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAoDoD","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 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, treestructureId } from './helperFunctions';\nimport { CommonTreeStructureProps, FolderType, TreeStructureType } from './types';\nimport ComboboxButton from './ComboboxButton';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\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 === '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`;\n\ninterface ScrollableDivProps {\n  type: TreeStructureType;\n}\n\nconst ScrollableDiv = styled.div<ScrollableDivProps>`\n  ${({ type }) =>\n    type === 'picker' &&\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\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: FolderType[];\n  label?: string;\n  maxLevel?: number;\n  onNewFolder?: (name: string, parentId: string) => Promise<IFolder>;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onNewFolder,\n  onSelectFolder,\n  targetResource,\n  type,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(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 [focusedFolder, setFocusedFolder] = useState<FolderType | undefined>();\n  const [selectedFolder, setSelectedFolder] = useState<FolderType | undefined>();\n  const [showTree, setShowTree] = useState(type === 'navigation');\n\n  const flattenedFolders = useMemo(() => flattenFolders(folders, openFolders), [folders, openFolders]);\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        if (type === 'picker') {\n          setFocusedFolder(selected);\n        }\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [defaultSelectedFolderId]);\n\n  const onToggleTree = (open: boolean) => {\n    setShowTree(open);\n    if (!open) {\n      setNewFolderParentId(undefined);\n    }\n  };\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        setFocusedFolder(closedFolder);\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        setSelectedFolder(newFolder);\n        onSelectFolder?.(newFolder.id);\n        setFocusedFolder(newFolder);\n        setOpenFolders(uniq(openFolders.concat(parentId)));\n        setNewFolderParentId?.(undefined);\n        ref.current?.focus();\n      }\n    });\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus();\n  };\n\n  const setFolderFocus = (folder: FolderType, focus?: boolean) => {\n    setFocusedFolder(folder);\n\n    if (focus) {\n      ref.current?.focus();\n    }\n  };\n\n  return (\n    <StyledTreeStructure>\n      {label && <StyledLabel id={treestructureId(type, 'label')}>{label}</StyledLabel>}\n      <TreeStructureWrapper\n        aria-label={label}\n        type={type}\n        onBlur={(e) => {\n          if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {\n            onToggleTree(false);\n          }\n        }}>\n        {type === 'picker' && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            focusedFolder={focusedFolder}\n            selectedFolder={selectedFolder}\n            setSelectedFolder={setSelectedFolder}\n            setFocusedFolder={setFocusedFolder}\n            onToggleTree={onToggleTree}\n            flattenedFolders={flattenedFolders}\n            onCloseFolder={onCloseFolder}\n            onOpenFolder={onOpenFolder}\n            onNewFolder={onNewFolder}\n            maxLevel={maxLevel}\n            setNewFolderParentId={setNewFolderParentId}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv type={type}>\n            <FolderItems\n              focusedFolder={focusedFolder}\n              folders={folders}\n              level={0}\n              loading={loading}\n              selectedFolder={selectedFolder}\n              maxLevel={maxLevel}\n              newFolderParentId={newFolderParentId}\n              onCancelNewFolder={onCancelNewFolder}\n              onCloseFolder={onCloseFolder}\n              onOpenFolder={onOpenFolder}\n              onSaveNewFolder={onSaveNewFolder}\n              onSelectFolder={onSelectFolder}\n              openFolders={openFolders}\n              setFocusedFolder={setFolderFocus}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */");
78
+
79
+ var TreeStructure = function TreeStructure(_ref3) {
80
+ var defaultOpenFolders = _ref3.defaultOpenFolders,
81
+ folders = _ref3.folders,
82
+ label = _ref3.label,
83
+ loading = _ref3.loading,
84
+ _ref3$maxLevel = _ref3.maxLevel,
85
+ maxLevel = _ref3$maxLevel === void 0 ? MAX_LEVEL_FOR_FOLDERS : _ref3$maxLevel,
86
+ onNewFolder = _ref3.onNewFolder,
87
+ onSelectFolder = _ref3.onSelectFolder,
88
+ targetResource = _ref3.targetResource,
89
+ type = _ref3.type;
147
90
  var ref = (0, _react.useRef)(null);
148
91
  var defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];
149
92
 
@@ -159,15 +102,15 @@ var TreeStructure = function TreeStructure(_ref4) {
159
102
 
160
103
  var _useState5 = (0, _react.useState)(),
161
104
  _useState6 = _slicedToArray(_useState5, 2),
162
- focusedId = _useState6[0],
163
- setFocusedId = _useState6[1];
105
+ focusedFolder = _useState6[0],
106
+ setFocusedFolder = _useState6[1];
164
107
 
165
108
  var _useState7 = (0, _react.useState)(),
166
109
  _useState8 = _slicedToArray(_useState7, 2),
167
110
  selectedFolder = _useState8[0],
168
111
  setSelectedFolder = _useState8[1];
169
112
 
170
- var _useState9 = (0, _react.useState)(type !== 'picker'),
113
+ var _useState9 = (0, _react.useState)(type === 'navigation'),
171
114
  _useState10 = _slicedToArray(_useState9, 2),
172
115
  showTree = _useState10[0],
173
116
  setShowTree = _useState10[1];
@@ -175,23 +118,6 @@ var TreeStructure = function TreeStructure(_ref4) {
175
118
  var flattenedFolders = (0, _react.useMemo)(function () {
176
119
  return (0, _helperFunctions.flattenFolders)(folders, openFolders);
177
120
  }, [folders, openFolders]);
178
- var visibleFolderIds = flattenedFolders.map(function (folder) {
179
- return folder.id;
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]);
195
121
  (0, _react.useEffect)(function () {
196
122
  if (defaultOpenFolders) {
197
123
  if (!defaultOpenFolders.every(function (element) {
@@ -204,9 +130,6 @@ var TreeStructure = function TreeStructure(_ref4) {
204
130
  } // eslint-disable-next-line react-hooks/exhaustive-deps
205
131
 
206
132
  }, [defaultOpenFolders]);
207
- (0, _react.useEffect)(function () {
208
- setNewFolderParentId(undefined);
209
- }, [selectedFolder]);
210
133
  (0, _react.useEffect)(function () {
211
134
  if (defaultSelectedFolderId !== undefined) {
212
135
  var selected = (0, _helperFunctions.flattenFolders)(folders).find(function (folder) {
@@ -215,15 +138,22 @@ var TreeStructure = function TreeStructure(_ref4) {
215
138
 
216
139
  if (selected) {
217
140
  setSelectedFolder(selected);
141
+
142
+ if (type === 'picker') {
143
+ setFocusedFolder(selected);
144
+ }
218
145
  }
219
146
  } // eslint-disable-next-line react-hooks/exhaustive-deps
220
147
 
221
148
  }, [defaultSelectedFolderId]);
222
- (0, _react.useEffect)(function () {
223
- if (!loading) {
149
+
150
+ var onToggleTree = function onToggleTree(open) {
151
+ setShowTree(open);
152
+
153
+ if (!open) {
224
154
  setNewFolderParentId(undefined);
225
155
  }
226
- }, [loading]);
156
+ };
227
157
 
228
158
  var onCloseFolder = function onCloseFolder(id) {
229
159
  var closedFolder = flattenedFolders.find(function (folder) {
@@ -236,12 +166,7 @@ var TreeStructure = function TreeStructure(_ref4) {
236
166
  if (subFolders.some(function (folder) {
237
167
  return folder.id === (selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.id);
238
168
  })) {
239
- if (onSelectFolder) {
240
- setSelectedFolder(closedFolder);
241
- onSelectFolder(closedFolder.id);
242
- }
243
-
244
- setFocusedId(closedFolder.id);
169
+ setFocusedFolder(closedFolder);
245
170
  }
246
171
  }
247
172
 
@@ -257,70 +182,70 @@ var TreeStructure = function TreeStructure(_ref4) {
257
182
  var onSaveNewFolder = function onSaveNewFolder(name, parentId) {
258
183
  onNewFolder === null || onNewFolder === void 0 ? void 0 : onNewFolder(name, parentId).then(function (newFolder) {
259
184
  if (newFolder) {
260
- setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
185
+ var _ref$current;
186
+
261
187
  setSelectedFolder(newFolder);
262
188
  onSelectFolder === null || onSelectFolder === void 0 ? void 0 : onSelectFolder(newFolder.id);
263
- setFocusedId(newFolder.id);
189
+ setFocusedFolder(newFolder);
264
190
  setOpenFolders((0, _lodash.uniq)(openFolders.concat(parentId)));
191
+ setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
192
+ (_ref$current = ref.current) === null || _ref$current === void 0 ? void 0 : _ref$current.focus();
265
193
  }
266
194
  });
267
195
  };
268
196
 
269
197
  var onCancelNewFolder = function onCancelNewFolder() {
198
+ var _ref$current2;
199
+
270
200
  setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
201
+ (_ref$current2 = ref.current) === null || _ref$current2 === void 0 ? void 0 : _ref$current2.focus();
271
202
  };
272
203
 
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);
204
+ var setFolderFocus = function setFolderFocus(folder, focus) {
205
+ setFocusedFolder(folder);
206
+
207
+ if (focus) {
208
+ var _ref$current3;
209
+
210
+ (_ref$current3 = ref.current) === null || _ref$current3 === void 0 ? void 0 : _ref$current3.focus();
302
211
  }
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);
212
+ };
213
+
214
+ return (0, _core2.jsx)(StyledTreeStructure, null, label && (0, _core2.jsx)(StyledLabel, {
215
+ id: (0, _helperFunctions.treestructureId)(type, 'label')
216
+ }, label), (0, _core2.jsx)(TreeStructureWrapper, {
217
+ "aria-label": label,
218
+ type: type,
219
+ onBlur: function onBlur(e) {
220
+ if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {
221
+ onToggleTree(false);
222
+ }
314
223
  }
315
- })), showTree && (0, _core2.jsx)(ScrollableDiv, {
224
+ }, type === 'picker' && (0, _core2.jsx)(_ComboboxButton["default"], {
225
+ ref: ref,
226
+ showTree: showTree,
227
+ type: type,
228
+ label: label,
229
+ focusedFolder: focusedFolder,
230
+ selectedFolder: selectedFolder,
231
+ setSelectedFolder: setSelectedFolder,
232
+ setFocusedFolder: setFocusedFolder,
233
+ onToggleTree: onToggleTree,
234
+ flattenedFolders: flattenedFolders,
235
+ onCloseFolder: onCloseFolder,
236
+ onOpenFolder: onOpenFolder,
237
+ onNewFolder: onNewFolder,
238
+ maxLevel: maxLevel,
239
+ setNewFolderParentId: setNewFolderParentId
240
+ }), showTree && (0, _core2.jsx)(ScrollableDiv, {
316
241
  type: type
317
242
  }, (0, _core2.jsx)(_FolderItems["default"], {
318
- focusedFolderId: focusedId,
243
+ focusedFolder: focusedFolder,
319
244
  folders: folders,
320
245
  level: 0,
321
246
  loading: loading,
322
247
  selectedFolder: selectedFolder,
323
- maxLevel: maximumLevelsOfFoldersAllowed,
248
+ maxLevel: maxLevel,
324
249
  newFolderParentId: newFolderParentId,
325
250
  onCancelNewFolder: onCancelNewFolder,
326
251
  onCloseFolder: onCloseFolder,
@@ -328,12 +253,14 @@ var TreeStructure = function TreeStructure(_ref4) {
328
253
  onSaveNewFolder: onSaveNewFolder,
329
254
  onSelectFolder: onSelectFolder,
330
255
  openFolders: openFolders,
331
- openOnFolderClick: openOnFolderClick,
332
- setFocusedId: setFocusedId,
256
+ setFocusedFolder: setFolderFocus,
333
257
  setSelectedFolder: setSelectedFolder,
334
258
  targetResource: targetResource,
335
- visibleFolders: visibleFolderIds,
336
- type: type
259
+ visibleFolders: flattenedFolders,
260
+ type: type,
261
+ closeTree: function closeTree() {
262
+ return onToggleTree(false);
263
+ }
337
264
  }))));
338
265
  };
339
266