@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
@@ -22,34 +22,22 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
22
22
  *
23
23
  */
24
24
  import React, { useEffect, useState, useMemo, useRef } from 'react';
25
- import { ButtonV2 as Button, IconButtonDualStates } from '@ndla/button';
26
- import { Plus } from '@ndla/icons/action';
27
- import { ChevronDown, ChevronUp } from '@ndla/icons/common';
28
- import Tooltip from '@ndla/tooltip';
29
- import { useTranslation } from 'react-i18next';
30
25
  import { colors, fonts, misc, spacing } from '@ndla/core';
31
26
  import { css } from '@emotion/core';
32
27
  import { uniq } from 'lodash';
33
28
  import FolderItems from './FolderItems';
34
- import { flattenFolders } from './helperFunctions';
29
+ import { flattenFolders, treestructureId } from './helperFunctions';
30
+ import ComboboxButton from './ComboboxButton';
35
31
  import { jsx as ___EmotionJSX } from "@emotion/core";
36
32
  export var MAX_LEVEL_FOR_FOLDERS = 4;
37
33
 
38
34
  var StyledLabel = _styled("label", {
39
35
  target: "e1dg1gdn0",
40
36
  label: "StyledLabel"
41
- })("font-weight:", 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"]} */"));
42
-
43
- var StyledRow = _styled("div", {
44
- target: "e1dg1gdn1",
45
- label: "StyledRow"
46
- })("display:flex;justify-content:space-between;padding:", spacing.xxsmall, ";border-bottom:", function (_ref) {
47
- var isOpen = _ref.isOpen;
48
- return isOpen && "1px solid ".concat(colors.brand.tertiary);
49
- }, ";" + (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"]} */"));
37
+ })("font-weight:", 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"]} */"));
50
38
 
51
39
  var StyledTreeStructure = _styled("div", {
52
- target: "e1dg1gdn2",
40
+ target: "e1dg1gdn1",
53
41
  label: "StyledTreeStructure"
54
42
  })(process.env.NODE_ENV === "production" ? {
55
43
  name: "k0sogd",
@@ -57,82 +45,37 @@ var StyledTreeStructure = _styled("div", {
57
45
  } : {
58
46
  name: "k0sogd",
59
47
  styles: "flex:1;display:flex;flex-direction:column;",
60
- 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"]} */",
48
+ 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"]} */",
61
49
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
62
50
  });
63
51
 
64
52
  var TreeStructureWrapper = _styled("div", {
65
- target: "e1dg1gdn3",
53
+ target: "e1dg1gdn2",
66
54
  label: "TreeStructureWrapper"
67
- })("display:flex;flex-direction:column;", function (_ref2) {
68
- var type = _ref2.type;
69
- return (type === 'normal' || type === 'picker') && /*#__PURE__*/css("overflow:hidden;border:1px solid ", colors.brand.neutral7, ";border-radius:", 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"]} */"));
70
- }, " transition:", misc.transition["default"], ";&:focus-within{border-color:", 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"]} */"));
55
+ })("display:flex;flex-direction:column;", function (_ref) {
56
+ var type = _ref.type;
57
+ return type === 'picker' && /*#__PURE__*/css("overflow:hidden;border:1px solid ", colors.brand.neutral7, ";border-radius:", 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"]} */"));
58
+ }, " transition:", misc.transition["default"], ";&:focus-within{border-color:", 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"]} */"));
71
59
 
72
60
  var ScrollableDiv = _styled("div", {
73
- target: "e1dg1gdn4",
61
+ target: "e1dg1gdn3",
74
62
  label: "ScrollableDiv"
75
- })(function (_ref3) {
76
- var type = _ref3.type;
77
- return (type === 'picker' || type === 'normal') && /*#__PURE__*/css("overflow:overlay;::-webkit-scrollbar{width:", spacing.small, ";}::-webkit-scrollbar-thumb{border:4px solid transparent;border-radius:14px;background-clip:padding-box;padding:0 4px;background-color:", 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"]} */"));
78
- }, 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"]} */");
79
-
80
- var StyledSelectedFolder = /*#__PURE__*/_styled(Button, {
81
- target: "e1dg1gdn5",
82
- label: "StyledSelectedFolder"
83
- })(process.env.NODE_ENV === "production" ? {
84
- name: "wj4dip",
85
- styles: "flex:1;justify-content:flex-start;:hover,:focus{background:none;box-shadow:none;border-color:transparent;}"
86
- } : {
87
- name: "wj4dip",
88
- styles: "flex:1;justify-content:flex-start;:hover,:focus{background:none;box-shadow:none;border-color:transparent;}",
89
- 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"]} */",
90
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
91
- });
92
-
93
- var StyledAddFolderButton = /*#__PURE__*/_styled(Button, {
94
- target: "e1dg1gdn6",
95
- label: "StyledAddFolderButton"
96
- })(process.env.NODE_ENV === "production" ? {
97
- name: "it7ogd",
98
- styles: "&,&:disabled{border-color:transparent;}"
99
- } : {
100
- name: "it7ogd",
101
- styles: "&,&:disabled{border-color:transparent;}",
102
- 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"]} */",
103
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
104
- });
105
-
106
- var StyledPlus = /*#__PURE__*/_styled(Plus, {
107
- target: "e1dg1gdn7",
108
- label: "StyledPlus"
109
- })(process.env.NODE_ENV === "production" ? {
110
- name: "1m01c8l",
111
- styles: "height:24px;width:24px;"
112
- } : {
113
- name: "1m01c8l",
114
- styles: "height:24px;width:24px;",
115
- 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"]} */",
116
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
117
- });
118
-
119
- var TreeStructure = function TreeStructure(_ref4) {
120
- var defaultOpenFolders = _ref4.defaultOpenFolders,
121
- folders = _ref4.folders,
122
- label = _ref4.label,
123
- loading = _ref4.loading,
124
- _ref4$maximumLevelsOf = _ref4.maximumLevelsOfFoldersAllowed,
125
- maximumLevelsOfFoldersAllowed = _ref4$maximumLevelsOf === void 0 ? MAX_LEVEL_FOR_FOLDERS : _ref4$maximumLevelsOf,
126
- onNewFolder = _ref4.onNewFolder,
127
- onSelectFolder = _ref4.onSelectFolder,
128
- openOnFolderClick = _ref4.openOnFolderClick,
129
- targetResource = _ref4.targetResource,
130
- _ref4$type = _ref4.type,
131
- type = _ref4$type === void 0 ? 'normal' : _ref4$type;
132
-
133
- var _useTranslation = useTranslation(),
134
- t = _useTranslation.t;
135
-
63
+ })(function (_ref2) {
64
+ var type = _ref2.type;
65
+ return type === 'picker' && /*#__PURE__*/css("overflow:overlay;::-webkit-scrollbar{width:", spacing.small, ";}::-webkit-scrollbar-thumb{border:4px solid transparent;border-radius:14px;background-clip:padding-box;padding:0 4px;background-color:", 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"]} */"));
66
+ }, 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"]} */");
67
+
68
+ var TreeStructure = function TreeStructure(_ref3) {
69
+ var defaultOpenFolders = _ref3.defaultOpenFolders,
70
+ folders = _ref3.folders,
71
+ label = _ref3.label,
72
+ loading = _ref3.loading,
73
+ _ref3$maxLevel = _ref3.maxLevel,
74
+ maxLevel = _ref3$maxLevel === void 0 ? MAX_LEVEL_FOR_FOLDERS : _ref3$maxLevel,
75
+ onNewFolder = _ref3.onNewFolder,
76
+ onSelectFolder = _ref3.onSelectFolder,
77
+ targetResource = _ref3.targetResource,
78
+ type = _ref3.type;
136
79
  var ref = useRef(null);
137
80
  var defaultSelectedFolderId = defaultOpenFolders && defaultOpenFolders[defaultOpenFolders.length - 1];
138
81
 
@@ -148,15 +91,15 @@ var TreeStructure = function TreeStructure(_ref4) {
148
91
 
149
92
  var _useState5 = useState(),
150
93
  _useState6 = _slicedToArray(_useState5, 2),
151
- focusedId = _useState6[0],
152
- setFocusedId = _useState6[1];
94
+ focusedFolder = _useState6[0],
95
+ setFocusedFolder = _useState6[1];
153
96
 
154
97
  var _useState7 = useState(),
155
98
  _useState8 = _slicedToArray(_useState7, 2),
156
99
  selectedFolder = _useState8[0],
157
100
  setSelectedFolder = _useState8[1];
158
101
 
159
- var _useState9 = useState(type !== 'picker'),
102
+ var _useState9 = useState(type === 'navigation'),
160
103
  _useState10 = _slicedToArray(_useState9, 2),
161
104
  showTree = _useState10[0],
162
105
  setShowTree = _useState10[1];
@@ -164,23 +107,6 @@ var TreeStructure = function TreeStructure(_ref4) {
164
107
  var flattenedFolders = useMemo(function () {
165
108
  return flattenFolders(folders, openFolders);
166
109
  }, [folders, openFolders]);
167
- var visibleFolderIds = flattenedFolders.map(function (folder) {
168
- return folder.id;
169
- });
170
- useEffect(function () {
171
- var handleClickOutside = function handleClickOutside(e) {
172
- if (e.target instanceof Element && ref.current && !ref.current.contains(e.target)) {
173
- setShowTree(false);
174
- }
175
- };
176
-
177
- if (type === 'picker') {
178
- document.addEventListener('mousedown', handleClickOutside);
179
- return function () {
180
- document.removeEventListener('mousedown', handleClickOutside);
181
- };
182
- }
183
- }, [ref, type]);
184
110
  useEffect(function () {
185
111
  if (defaultOpenFolders) {
186
112
  if (!defaultOpenFolders.every(function (element) {
@@ -193,9 +119,6 @@ var TreeStructure = function TreeStructure(_ref4) {
193
119
  } // eslint-disable-next-line react-hooks/exhaustive-deps
194
120
 
195
121
  }, [defaultOpenFolders]);
196
- useEffect(function () {
197
- setNewFolderParentId(undefined);
198
- }, [selectedFolder]);
199
122
  useEffect(function () {
200
123
  if (defaultSelectedFolderId !== undefined) {
201
124
  var selected = flattenFolders(folders).find(function (folder) {
@@ -204,15 +127,22 @@ var TreeStructure = function TreeStructure(_ref4) {
204
127
 
205
128
  if (selected) {
206
129
  setSelectedFolder(selected);
130
+
131
+ if (type === 'picker') {
132
+ setFocusedFolder(selected);
133
+ }
207
134
  }
208
135
  } // eslint-disable-next-line react-hooks/exhaustive-deps
209
136
 
210
137
  }, [defaultSelectedFolderId]);
211
- useEffect(function () {
212
- if (!loading) {
138
+
139
+ var onToggleTree = function onToggleTree(open) {
140
+ setShowTree(open);
141
+
142
+ if (!open) {
213
143
  setNewFolderParentId(undefined);
214
144
  }
215
- }, [loading]);
145
+ };
216
146
 
217
147
  var onCloseFolder = function onCloseFolder(id) {
218
148
  var closedFolder = flattenedFolders.find(function (folder) {
@@ -225,12 +155,7 @@ var TreeStructure = function TreeStructure(_ref4) {
225
155
  if (subFolders.some(function (folder) {
226
156
  return folder.id === (selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.id);
227
157
  })) {
228
- if (onSelectFolder) {
229
- setSelectedFolder(closedFolder);
230
- onSelectFolder(closedFolder.id);
231
- }
232
-
233
- setFocusedId(closedFolder.id);
158
+ setFocusedFolder(closedFolder);
234
159
  }
235
160
  }
236
161
 
@@ -246,70 +171,70 @@ var TreeStructure = function TreeStructure(_ref4) {
246
171
  var onSaveNewFolder = function onSaveNewFolder(name, parentId) {
247
172
  onNewFolder === null || onNewFolder === void 0 ? void 0 : onNewFolder(name, parentId).then(function (newFolder) {
248
173
  if (newFolder) {
249
- setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
174
+ var _ref$current;
175
+
250
176
  setSelectedFolder(newFolder);
251
177
  onSelectFolder === null || onSelectFolder === void 0 ? void 0 : onSelectFolder(newFolder.id);
252
- setFocusedId(newFolder.id);
178
+ setFocusedFolder(newFolder);
253
179
  setOpenFolders(uniq(openFolders.concat(parentId)));
180
+ setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
181
+ (_ref$current = ref.current) === null || _ref$current === void 0 ? void 0 : _ref$current.focus();
254
182
  }
255
183
  });
256
184
  };
257
185
 
258
186
  var onCancelNewFolder = function onCancelNewFolder() {
187
+ var _ref$current2;
188
+
259
189
  setNewFolderParentId === null || setNewFolderParentId === void 0 ? void 0 : setNewFolderParentId(undefined);
190
+ (_ref$current2 = ref.current) === null || _ref$current2 === void 0 ? void 0 : _ref$current2.focus();
260
191
  };
261
192
 
262
- var canAddFolder = selectedFolder && (selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.breadcrumbs.length) < (maximumLevelsOfFoldersAllowed || 1);
263
- return ___EmotionJSX(StyledTreeStructure, {
264
- ref: ref
265
- }, label && ___EmotionJSX(StyledLabel, null, label), ___EmotionJSX(TreeStructureWrapper, {
266
- "aria-label": label,
267
- type: type
268
- }, type === 'picker' && ___EmotionJSX(StyledRow, {
269
- isOpen: showTree
270
- }, ___EmotionJSX(StyledSelectedFolder, {
271
- variant: "ghost",
272
- colorTheme: "light",
273
- fontWeight: "normal",
274
- shape: "sharp",
275
- onClick: function onClick() {
276
- setShowTree(!showTree);
277
- }
278
- }, selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.name), onNewFolder && showTree && ___EmotionJSX(Tooltip, {
279
- tooltip: canAddFolder ? t('myNdla.newFolderUnder', {
280
- folderName: selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.name
281
- }) : t('treeStructure.maxFoldersAlreadyAdded')
282
- }, ___EmotionJSX(StyledAddFolderButton, {
283
- variant: "outline",
284
- shape: "pill",
285
- disabled: !canAddFolder,
286
- "aria-label": canAddFolder ? t('myNdla.newFolderUnder', {
287
- folderName: selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.name
288
- }) : t('treeStructure.maxFoldersAlreadyAdded'),
289
- onClick: function onClick() {
290
- return setNewFolderParentId(selectedFolder === null || selectedFolder === void 0 ? void 0 : selectedFolder.id);
193
+ var setFolderFocus = function setFolderFocus(folder, focus) {
194
+ setFocusedFolder(folder);
195
+
196
+ if (focus) {
197
+ var _ref$current3;
198
+
199
+ (_ref$current3 = ref.current) === null || _ref$current3 === void 0 ? void 0 : _ref$current3.focus();
291
200
  }
292
- }, ___EmotionJSX(StyledPlus, null), " ", t('myNdla.newFolder'))), ___EmotionJSX(IconButtonDualStates, {
293
- ariaLabelActive: t('treeStructure.hideFolders'),
294
- ariaLabelInActive: t('treeStructure.showFolders'),
295
- active: showTree,
296
- variant: "ghost",
297
- colorTheme: "greyLighter",
298
- inactiveIcon: ___EmotionJSX(ChevronDown, null),
299
- activeIcon: ___EmotionJSX(ChevronUp, null),
300
- size: "small",
301
- onClick: function onClick() {
302
- setShowTree(!showTree);
201
+ };
202
+
203
+ return ___EmotionJSX(StyledTreeStructure, null, label && ___EmotionJSX(StyledLabel, {
204
+ id: treestructureId(type, 'label')
205
+ }, label), ___EmotionJSX(TreeStructureWrapper, {
206
+ "aria-label": label,
207
+ type: type,
208
+ onBlur: function onBlur(e) {
209
+ if (type === 'picker' && !e.currentTarget.contains(e.relatedTarget)) {
210
+ onToggleTree(false);
211
+ }
303
212
  }
304
- })), showTree && ___EmotionJSX(ScrollableDiv, {
213
+ }, type === 'picker' && ___EmotionJSX(ComboboxButton, {
214
+ ref: ref,
215
+ showTree: showTree,
216
+ type: type,
217
+ label: label,
218
+ focusedFolder: focusedFolder,
219
+ selectedFolder: selectedFolder,
220
+ setSelectedFolder: setSelectedFolder,
221
+ setFocusedFolder: setFocusedFolder,
222
+ onToggleTree: onToggleTree,
223
+ flattenedFolders: flattenedFolders,
224
+ onCloseFolder: onCloseFolder,
225
+ onOpenFolder: onOpenFolder,
226
+ onNewFolder: onNewFolder,
227
+ maxLevel: maxLevel,
228
+ setNewFolderParentId: setNewFolderParentId
229
+ }), showTree && ___EmotionJSX(ScrollableDiv, {
305
230
  type: type
306
231
  }, ___EmotionJSX(FolderItems, {
307
- focusedFolderId: focusedId,
232
+ focusedFolder: focusedFolder,
308
233
  folders: folders,
309
234
  level: 0,
310
235
  loading: loading,
311
236
  selectedFolder: selectedFolder,
312
- maxLevel: maximumLevelsOfFoldersAllowed,
237
+ maxLevel: maxLevel,
313
238
  newFolderParentId: newFolderParentId,
314
239
  onCancelNewFolder: onCancelNewFolder,
315
240
  onCloseFolder: onCloseFolder,
@@ -317,12 +242,14 @@ var TreeStructure = function TreeStructure(_ref4) {
317
242
  onSaveNewFolder: onSaveNewFolder,
318
243
  onSelectFolder: onSelectFolder,
319
244
  openFolders: openFolders,
320
- openOnFolderClick: openOnFolderClick,
321
- setFocusedId: setFocusedId,
245
+ setFocusedFolder: setFolderFocus,
322
246
  setSelectedFolder: setSelectedFolder,
323
247
  targetResource: targetResource,
324
- visibleFolders: visibleFolderIds,
325
- type: type
248
+ visibleFolders: flattenedFolders,
249
+ type: type,
250
+ closeTree: function closeTree() {
251
+ return onToggleTree(false);
252
+ }
326
253
  }))));
327
254
  };
328
255