@ndla/ui 56.0.13-alpha.0 → 56.0.15-alpha.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/dist/panda.buildinfo.json +7 -2
  2. package/dist/styles.css +24 -4
  3. package/es/Article/Article.js +4 -1
  4. package/es/ContentTypeBadge/ContentTypeBadge.js +2 -0
  5. package/es/TreeStructure/TreeStructure.js +292 -181
  6. package/es/TreeStructure/helperFunctions.js +0 -3
  7. package/es/TreeStructure/index.js +1 -2
  8. package/es/index.js +0 -1
  9. package/es/locale/messages-en.js +1 -1
  10. package/es/locale/messages-nb.js +1 -1
  11. package/es/locale/messages-nn.js +1 -1
  12. package/es/locale/messages-se.js +1 -1
  13. package/es/locale/messages-sma.js +1 -1
  14. package/es/styles.css +24 -4
  15. package/lib/Article/Article.js +4 -1
  16. package/lib/ContentTypeBadge/ContentTypeBadge.js +2 -0
  17. package/lib/TreeStructure/TreeStructure.d.ts +7 -6
  18. package/lib/TreeStructure/TreeStructure.js +293 -180
  19. package/lib/TreeStructure/helperFunctions.d.ts +0 -2
  20. package/lib/TreeStructure/helperFunctions.js +2 -6
  21. package/lib/TreeStructure/index.d.ts +1 -2
  22. package/lib/TreeStructure/index.js +2 -3
  23. package/lib/TreeStructure/types.d.ts +4 -22
  24. package/lib/index.d.ts +0 -2
  25. package/lib/index.js +0 -7
  26. package/lib/locale/messages-en.js +1 -1
  27. package/lib/locale/messages-nb.js +1 -1
  28. package/lib/locale/messages-nn.js +1 -1
  29. package/lib/locale/messages-se.js +1 -1
  30. package/lib/locale/messages-sma.js +1 -1
  31. package/lib/styles.css +24 -4
  32. package/package.json +7 -8
  33. package/src/Article/Article.tsx +4 -1
  34. package/src/ContentTypeBadge/ContentTypeBadge.tsx +2 -0
  35. package/src/TreeStructure/TreeStructure.stories.tsx +38 -68
  36. package/src/TreeStructure/TreeStructure.tsx +307 -194
  37. package/src/TreeStructure/helperFunctions.ts +0 -5
  38. package/src/TreeStructure/index.ts +1 -2
  39. package/src/TreeStructure/types.ts +4 -25
  40. package/src/index.ts +0 -3
  41. package/src/locale/messages-en.ts +1 -1
  42. package/src/locale/messages-nb.ts +1 -1
  43. package/src/locale/messages-nn.ts +1 -1
  44. package/src/locale/messages-se.ts +1 -1
  45. package/src/locale/messages-sma.ts +1 -1
  46. package/es/ProgrammeCard/ProgrammeCard.js +0 -51
  47. package/es/ProgrammeCard/index.js +0 -9
  48. package/es/TreeStructure/AddFolderButton.js +0 -80
  49. package/es/TreeStructure/ComboboxButton.js +0 -127
  50. package/es/TreeStructure/FolderItem.js +0 -225
  51. package/es/TreeStructure/FolderItems.js +0 -95
  52. package/es/TreeStructure/arrowNavigation.js +0 -35
  53. package/lib/ProgrammeCard/ProgrammeCard.d.ts +0 -23
  54. package/lib/ProgrammeCard/ProgrammeCard.js +0 -58
  55. package/lib/ProgrammeCard/index.d.ts +0 -9
  56. package/lib/ProgrammeCard/index.js +0 -13
  57. package/lib/TreeStructure/AddFolderButton.d.ts +0 -17
  58. package/lib/TreeStructure/AddFolderButton.js +0 -85
  59. package/lib/TreeStructure/ComboboxButton.d.ts +0 -27
  60. package/lib/TreeStructure/ComboboxButton.js +0 -134
  61. package/lib/TreeStructure/FolderItem.d.ts +0 -17
  62. package/lib/TreeStructure/FolderItem.js +0 -230
  63. package/lib/TreeStructure/FolderItems.d.ts +0 -22
  64. package/lib/TreeStructure/FolderItems.js +0 -100
  65. package/lib/TreeStructure/arrowNavigation.d.ts +0 -10
  66. package/lib/TreeStructure/arrowNavigation.js +0 -42
  67. package/src/ProgrammeCard/ProgrammeCard.stories.tsx +0 -35
  68. package/src/ProgrammeCard/ProgrammeCard.tsx +0 -78
  69. package/src/ProgrammeCard/index.tsx +0 -10
  70. package/src/TreeStructure/AddFolderButton.tsx +0 -79
  71. package/src/TreeStructure/ComboboxButton.tsx +0 -172
  72. package/src/TreeStructure/FolderItem.tsx +0 -307
  73. package/src/TreeStructure/FolderItems.tsx +0 -121
  74. package/src/TreeStructure/arrowNavigation.ts +0 -54
@@ -3,212 +3,325 @@
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- exports.default = exports.MAX_LEVEL_FOR_FOLDERS = void 0;
7
- var _base = _interopRequireDefault(require("@emotion/styled/base"));
6
+ exports.TreeStructure = exports.MAX_LEVEL_FOR_FOLDERS = void 0;
8
7
  var _react = require("react");
9
- var _core = require("@ndla/core");
10
- var _AddFolderButton = _interopRequireDefault(require("./AddFolderButton"));
11
- var _ComboboxButton = _interopRequireDefault(require("./ComboboxButton"));
12
- var _FolderItems = _interopRequireDefault(require("./FolderItems"));
8
+ var _reactDom = require("react-dom");
9
+ var _reactI18next = require("react-i18next");
10
+ var _react2 = require("@ark-ui/react");
11
+ var _action = require("@ndla/icons/action");
12
+ var _common = require("@ndla/icons/common");
13
+ var _contentType = require("@ndla/icons/contentType");
14
+ var _editor = require("@ndla/icons/editor");
15
+ var _primitives = require("@ndla/primitives");
16
+ var _jsx2 = require("@ndla/styled-system/jsx");
13
17
  var _helperFunctions = require("./helperFunctions");
14
18
  var _jsxRuntime = require("react/jsx-runtime");
15
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
16
- function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; } /**
17
- * Copyright (c) 2022-present, NDLA.
19
+ /**
20
+ * Copyright (c) 2024-present, NDLA.
18
21
  *
19
22
  * This source code is licensed under the GPLv3 license found in the
20
23
  * LICENSE file in the root directory of this source tree.
21
24
  *
22
25
  */
26
+
23
27
  const MAX_LEVEL_FOR_FOLDERS = exports.MAX_LEVEL_FOR_FOLDERS = 5;
24
- const uniq = array => Array.from(new Set(array));
25
- const StyledLabel = /*#__PURE__*/(0, _base.default)("label", {
26
- target: "e1dg1gdn4",
27
- label: "StyledLabel"
28
- })("font-weight:", _core.fonts.weight.semibold, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAsBgC","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 { useEffect, useState, useMemo, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { colors, fonts, misc, utils } from \"@ndla/core\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport AddFolderButton from \"./AddFolderButton\";\nimport ComboboxButton from \"./ComboboxButton\";\nimport FolderItems from \"./FolderItems\";\nimport { flattenFolders, treestructureId } from \"./helperFunctions\";\nimport { CommonTreeStructureProps, NewFolderInputFunc } from \"./types\";\n\nexport const MAX_LEVEL_FOR_FOLDERS = 5;\n\nconst uniq = <T,>(array: T[]): T[] => Array.from(new Set(array));\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 Row = styled.div`\n  display: flex;\n  align-items: center;\n  justify-content: space-between;\n`;\n\nconst TreeStructureWrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  transition: ${misc.transition.default};\n  &[data-type=\"picker\"] {\n    overflow: hidden;\n    border: 1px solid ${colors.brand.neutral7};\n    border-radius: ${misc.borderRadius};\n    scroll-behavior: smooth;\n  }\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\n\nconst ScrollableDiv = styled.div`\n  &[data-type=\"picker\"] {\n    overflow: auto;\n    overflow: overlay;\n    ${utils.scrollbar}\n  }\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: IFolder[];\n  label?: string;\n  maxLevel?: number;\n  newFolderInput?: NewFolderInputFunc;\n  onSelectFolder?: (id: string) => void;\n  ariaDescribedby?: string;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onSelectFolder,\n  targetResource,\n  type,\n  newFolderInput,\n  ariaDescribedby,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(null);\n\n  const defaultSelectedFolderId = 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<IFolder | undefined>();\n  const [selectedFolder, _setSelectedFolder] = useState<IFolder | 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 setSelectedFolder = (folder: IFolder) => {\n    _setSelectedFolder(folder);\n    onSelectFolder?.(folder.id);\n  };\n\n  const setFocusedFolder = (folder: IFolder) => {\n    _setFocusedFolder(folder);\n    setNewFolderParentId(undefined);\n\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\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 onNewFolderCreated = (newFolder: IFolder | undefined, parentId: string) => {\n    if (newFolder) {\n      setSelectedFolder(newFolder);\n      setFocusedFolder(newFolder);\n      setOpenFolders(uniq(openFolders.concat(parentId)));\n      setNewFolderParentId?.(undefined);\n      ref.current?.focus({ preventScroll: true });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maxLevel || 1);\n\n  return (\n    <StyledTreeStructure\n      onBlur={(e) => {\n        if (type === \"picker\" && !e.currentTarget.contains(e.relatedTarget)) {\n          onToggleTree(false);\n        }\n      }}\n    >\n      <Row>\n        {label && <StyledLabel id={treestructureId(type, \"label\")}>{label}</StyledLabel>}\n        {type === \"picker\" && (\n          <AddFolderButton\n            loading={loading}\n            canAddFolder={!!canAddFolder}\n            focusedFolder={focusedFolder}\n            setNewFolderParentId={setNewFolderParentId}\n            setShowTree={setShowTree}\n          />\n        )}\n      </Row>\n      <TreeStructureWrapper aria-label={label} data-type={type}>\n        {type === \"picker\" && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            loading={loading}\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            ariaDescribedby={ariaDescribedby}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv data-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              openFolders={openFolders}\n              setFocusedFolder={setFocusedFolder}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n              newFolderInput={newFolderInput}\n              onCreate={onNewFolderCreated}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
29
- const StyledTreeStructure = /*#__PURE__*/(0, _base.default)("div", {
30
- target: "e1dg1gdn3",
31
- label: "StyledTreeStructure"
32
- })(process.env.NODE_ENV === "production" ? {
33
- name: "1t91kdm",
34
- styles: "flex:1;display:flex;flex-direction:column"
35
- } : {
36
- name: "1t91kdm",
37
- styles: "flex:1;display:flex;flex-direction:column",
38
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA0BsC","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 { useEffect, useState, useMemo, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { colors, fonts, misc, utils } from \"@ndla/core\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport AddFolderButton from \"./AddFolderButton\";\nimport ComboboxButton from \"./ComboboxButton\";\nimport FolderItems from \"./FolderItems\";\nimport { flattenFolders, treestructureId } from \"./helperFunctions\";\nimport { CommonTreeStructureProps, NewFolderInputFunc } from \"./types\";\n\nexport const MAX_LEVEL_FOR_FOLDERS = 5;\n\nconst uniq = <T,>(array: T[]): T[] => Array.from(new Set(array));\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 Row = styled.div`\n  display: flex;\n  align-items: center;\n  justify-content: space-between;\n`;\n\nconst TreeStructureWrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  transition: ${misc.transition.default};\n  &[data-type=\"picker\"] {\n    overflow: hidden;\n    border: 1px solid ${colors.brand.neutral7};\n    border-radius: ${misc.borderRadius};\n    scroll-behavior: smooth;\n  }\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\n\nconst ScrollableDiv = styled.div`\n  &[data-type=\"picker\"] {\n    overflow: auto;\n    overflow: overlay;\n    ${utils.scrollbar}\n  }\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: IFolder[];\n  label?: string;\n  maxLevel?: number;\n  newFolderInput?: NewFolderInputFunc;\n  onSelectFolder?: (id: string) => void;\n  ariaDescribedby?: string;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onSelectFolder,\n  targetResource,\n  type,\n  newFolderInput,\n  ariaDescribedby,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(null);\n\n  const defaultSelectedFolderId = 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<IFolder | undefined>();\n  const [selectedFolder, _setSelectedFolder] = useState<IFolder | 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 setSelectedFolder = (folder: IFolder) => {\n    _setSelectedFolder(folder);\n    onSelectFolder?.(folder.id);\n  };\n\n  const setFocusedFolder = (folder: IFolder) => {\n    _setFocusedFolder(folder);\n    setNewFolderParentId(undefined);\n\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\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 onNewFolderCreated = (newFolder: IFolder | undefined, parentId: string) => {\n    if (newFolder) {\n      setSelectedFolder(newFolder);\n      setFocusedFolder(newFolder);\n      setOpenFolders(uniq(openFolders.concat(parentId)));\n      setNewFolderParentId?.(undefined);\n      ref.current?.focus({ preventScroll: true });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maxLevel || 1);\n\n  return (\n    <StyledTreeStructure\n      onBlur={(e) => {\n        if (type === \"picker\" && !e.currentTarget.contains(e.relatedTarget)) {\n          onToggleTree(false);\n        }\n      }}\n    >\n      <Row>\n        {label && <StyledLabel id={treestructureId(type, \"label\")}>{label}</StyledLabel>}\n        {type === \"picker\" && (\n          <AddFolderButton\n            loading={loading}\n            canAddFolder={!!canAddFolder}\n            focusedFolder={focusedFolder}\n            setNewFolderParentId={setNewFolderParentId}\n            setShowTree={setShowTree}\n          />\n        )}\n      </Row>\n      <TreeStructureWrapper aria-label={label} data-type={type}>\n        {type === \"picker\" && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            loading={loading}\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            ariaDescribedby={ariaDescribedby}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv data-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              openFolders={openFolders}\n              setFocusedFolder={setFocusedFolder}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n              newFolderInput={newFolderInput}\n              onCreate={onNewFolderCreated}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */",
39
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
28
+ const StyledButton = (0, _jsx2.styled)(_primitives.Button, {
29
+ base: {
30
+ width: "100%",
31
+ justifyContent: "space-between",
32
+ "& span": {
33
+ overflow: "hidden",
34
+ textOverflow: "ellipsis"
35
+ }
36
+ }
37
+ });
38
+ const StyledHStack = (0, _jsx2.styled)(_jsx2.HStack, {
39
+ base: {
40
+ overflow: "hidden"
41
+ }
42
+ });
43
+ const StyledHeartFill = (0, _jsx2.styled)(_action.HeartFill, {
44
+ base: {
45
+ color: "icon.strong"
46
+ }
47
+ });
48
+ const StyledFolderLine = (0, _jsx2.styled)(_editor.FolderLine, {
49
+ base: {
50
+ color: "icon.strong"
51
+ }
40
52
  });
41
- const Row = /*#__PURE__*/(0, _base.default)("div", {
42
- target: "e1dg1gdn2",
43
- label: "Row"
44
- })(process.env.NODE_ENV === "production" ? {
45
- name: "bcffy2",
46
- styles: "display:flex;align-items:center;justify-content:space-between"
47
- } : {
48
- name: "bcffy2",
49
- styles: "display:flex;align-items:center;justify-content:space-between",
50
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAgCsB","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 { useEffect, useState, useMemo, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { colors, fonts, misc, utils } from \"@ndla/core\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport AddFolderButton from \"./AddFolderButton\";\nimport ComboboxButton from \"./ComboboxButton\";\nimport FolderItems from \"./FolderItems\";\nimport { flattenFolders, treestructureId } from \"./helperFunctions\";\nimport { CommonTreeStructureProps, NewFolderInputFunc } from \"./types\";\n\nexport const MAX_LEVEL_FOR_FOLDERS = 5;\n\nconst uniq = <T,>(array: T[]): T[] => Array.from(new Set(array));\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 Row = styled.div`\n  display: flex;\n  align-items: center;\n  justify-content: space-between;\n`;\n\nconst TreeStructureWrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  transition: ${misc.transition.default};\n  &[data-type=\"picker\"] {\n    overflow: hidden;\n    border: 1px solid ${colors.brand.neutral7};\n    border-radius: ${misc.borderRadius};\n    scroll-behavior: smooth;\n  }\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\n\nconst ScrollableDiv = styled.div`\n  &[data-type=\"picker\"] {\n    overflow: auto;\n    overflow: overlay;\n    ${utils.scrollbar}\n  }\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: IFolder[];\n  label?: string;\n  maxLevel?: number;\n  newFolderInput?: NewFolderInputFunc;\n  onSelectFolder?: (id: string) => void;\n  ariaDescribedby?: string;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onSelectFolder,\n  targetResource,\n  type,\n  newFolderInput,\n  ariaDescribedby,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(null);\n\n  const defaultSelectedFolderId = 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<IFolder | undefined>();\n  const [selectedFolder, _setSelectedFolder] = useState<IFolder | 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 setSelectedFolder = (folder: IFolder) => {\n    _setSelectedFolder(folder);\n    onSelectFolder?.(folder.id);\n  };\n\n  const setFocusedFolder = (folder: IFolder) => {\n    _setFocusedFolder(folder);\n    setNewFolderParentId(undefined);\n\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\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 onNewFolderCreated = (newFolder: IFolder | undefined, parentId: string) => {\n    if (newFolder) {\n      setSelectedFolder(newFolder);\n      setFocusedFolder(newFolder);\n      setOpenFolders(uniq(openFolders.concat(parentId)));\n      setNewFolderParentId?.(undefined);\n      ref.current?.focus({ preventScroll: true });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maxLevel || 1);\n\n  return (\n    <StyledTreeStructure\n      onBlur={(e) => {\n        if (type === \"picker\" && !e.currentTarget.contains(e.relatedTarget)) {\n          onToggleTree(false);\n        }\n      }}\n    >\n      <Row>\n        {label && <StyledLabel id={treestructureId(type, \"label\")}>{label}</StyledLabel>}\n        {type === \"picker\" && (\n          <AddFolderButton\n            loading={loading}\n            canAddFolder={!!canAddFolder}\n            focusedFolder={focusedFolder}\n            setNewFolderParentId={setNewFolderParentId}\n            setShowTree={setShowTree}\n          />\n        )}\n      </Row>\n      <TreeStructureWrapper aria-label={label} data-type={type}>\n        {type === \"picker\" && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            loading={loading}\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            ariaDescribedby={ariaDescribedby}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv data-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              openFolders={openFolders}\n              setFocusedFolder={setFocusedFolder}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n              newFolderInput={newFolderInput}\n              onCreate={onNewFolderCreated}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */",
51
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
53
+ const StyledFolderUserLine = (0, _jsx2.styled)(_contentType.FolderUserLine, {
54
+ base: {
55
+ color: "icon.strong"
56
+ }
57
+ });
58
+ const StyledTreeRootProvider = (0, _jsx2.styled)(_primitives.TreeRootProvider, {
59
+ base: {
60
+ width: "100%",
61
+ maxHeight: "inherit"
62
+ }
63
+ });
64
+ const StyledPopoverContent = (0, _jsx2.styled)(_primitives.PopoverContent, {
65
+ base: {
66
+ display: "flex",
67
+ flexDirection: "column",
68
+ gap: "xsmall",
69
+ overflow: "auto",
70
+ maxHeight: "inherit",
71
+ paddingInline: "xsmall",
72
+ paddingBlock: "xsmall"
73
+ }
74
+ });
75
+ const LabelHStack = (0, _jsx2.styled)(_jsx2.HStack, {
76
+ base: {
77
+ width: "100%"
78
+ }
52
79
  });
53
- const TreeStructureWrapper = /*#__PURE__*/(0, _base.default)("div", {
54
- target: "e1dg1gdn1",
55
- label: "TreeStructureWrapper"
56
- })("display:flex;flex-direction:column;transition:", _core.misc.transition.default, ";&[data-type=\"picker\"]{overflow:hidden;border:1px solid ", _core.colors.brand.neutral7, ";border-radius:", _core.misc.borderRadius, ";scroll-behavior:smooth;}&:focus-within{border-color:", _core.colors.brand.tertiary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAsCuC","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 { useEffect, useState, useMemo, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { colors, fonts, misc, utils } from \"@ndla/core\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport AddFolderButton from \"./AddFolderButton\";\nimport ComboboxButton from \"./ComboboxButton\";\nimport FolderItems from \"./FolderItems\";\nimport { flattenFolders, treestructureId } from \"./helperFunctions\";\nimport { CommonTreeStructureProps, NewFolderInputFunc } from \"./types\";\n\nexport const MAX_LEVEL_FOR_FOLDERS = 5;\n\nconst uniq = <T,>(array: T[]): T[] => Array.from(new Set(array));\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 Row = styled.div`\n  display: flex;\n  align-items: center;\n  justify-content: space-between;\n`;\n\nconst TreeStructureWrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  transition: ${misc.transition.default};\n  &[data-type=\"picker\"] {\n    overflow: hidden;\n    border: 1px solid ${colors.brand.neutral7};\n    border-radius: ${misc.borderRadius};\n    scroll-behavior: smooth;\n  }\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\n\nconst ScrollableDiv = styled.div`\n  &[data-type=\"picker\"] {\n    overflow: auto;\n    overflow: overlay;\n    ${utils.scrollbar}\n  }\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: IFolder[];\n  label?: string;\n  maxLevel?: number;\n  newFolderInput?: NewFolderInputFunc;\n  onSelectFolder?: (id: string) => void;\n  ariaDescribedby?: string;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onSelectFolder,\n  targetResource,\n  type,\n  newFolderInput,\n  ariaDescribedby,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(null);\n\n  const defaultSelectedFolderId = 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<IFolder | undefined>();\n  const [selectedFolder, _setSelectedFolder] = useState<IFolder | 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 setSelectedFolder = (folder: IFolder) => {\n    _setSelectedFolder(folder);\n    onSelectFolder?.(folder.id);\n  };\n\n  const setFocusedFolder = (folder: IFolder) => {\n    _setFocusedFolder(folder);\n    setNewFolderParentId(undefined);\n\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\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 onNewFolderCreated = (newFolder: IFolder | undefined, parentId: string) => {\n    if (newFolder) {\n      setSelectedFolder(newFolder);\n      setFocusedFolder(newFolder);\n      setOpenFolders(uniq(openFolders.concat(parentId)));\n      setNewFolderParentId?.(undefined);\n      ref.current?.focus({ preventScroll: true });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maxLevel || 1);\n\n  return (\n    <StyledTreeStructure\n      onBlur={(e) => {\n        if (type === \"picker\" && !e.currentTarget.contains(e.relatedTarget)) {\n          onToggleTree(false);\n        }\n      }}\n    >\n      <Row>\n        {label && <StyledLabel id={treestructureId(type, \"label\")}>{label}</StyledLabel>}\n        {type === \"picker\" && (\n          <AddFolderButton\n            loading={loading}\n            canAddFolder={!!canAddFolder}\n            focusedFolder={focusedFolder}\n            setNewFolderParentId={setNewFolderParentId}\n            setShowTree={setShowTree}\n          />\n        )}\n      </Row>\n      <TreeStructureWrapper aria-label={label} data-type={type}>\n        {type === \"picker\" && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            loading={loading}\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            ariaDescribedby={ariaDescribedby}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv data-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              openFolders={openFolders}\n              setFocusedFolder={setFocusedFolder}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n              newFolderInput={newFolderInput}\n              onCreate={onNewFolderCreated}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
57
- const ScrollableDiv = /*#__PURE__*/(0, _base.default)("div", {
58
- target: "e1dg1gdn0",
59
- label: "ScrollableDiv"
60
- })("&[data-type=\"picker\"]{overflow:auto;overflow:overlay;", _core.utils.scrollbar, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AAqDgC","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 { useEffect, useState, useMemo, useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { colors, fonts, misc, utils } from \"@ndla/core\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport AddFolderButton from \"./AddFolderButton\";\nimport ComboboxButton from \"./ComboboxButton\";\nimport FolderItems from \"./FolderItems\";\nimport { flattenFolders, treestructureId } from \"./helperFunctions\";\nimport { CommonTreeStructureProps, NewFolderInputFunc } from \"./types\";\n\nexport const MAX_LEVEL_FOR_FOLDERS = 5;\n\nconst uniq = <T,>(array: T[]): T[] => Array.from(new Set(array));\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 Row = styled.div`\n  display: flex;\n  align-items: center;\n  justify-content: space-between;\n`;\n\nconst TreeStructureWrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  transition: ${misc.transition.default};\n  &[data-type=\"picker\"] {\n    overflow: hidden;\n    border: 1px solid ${colors.brand.neutral7};\n    border-radius: ${misc.borderRadius};\n    scroll-behavior: smooth;\n  }\n  &:focus-within {\n    border-color: ${colors.brand.tertiary};\n  }\n`;\n\nconst ScrollableDiv = styled.div`\n  &[data-type=\"picker\"] {\n    overflow: auto;\n    overflow: overlay;\n    ${utils.scrollbar}\n  }\n`;\n\nexport interface TreeStructureProps extends CommonTreeStructureProps {\n  defaultOpenFolders?: string[];\n  folders: IFolder[];\n  label?: string;\n  maxLevel?: number;\n  newFolderInput?: NewFolderInputFunc;\n  onSelectFolder?: (id: string) => void;\n  ariaDescribedby?: string;\n}\n\nconst TreeStructure = ({\n  defaultOpenFolders,\n  folders,\n  label,\n  loading,\n  maxLevel = MAX_LEVEL_FOR_FOLDERS,\n  onSelectFolder,\n  targetResource,\n  type,\n  newFolderInput,\n  ariaDescribedby,\n}: TreeStructureProps) => {\n  const ref = useRef<HTMLButtonElement>(null);\n\n  const defaultSelectedFolderId = 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<IFolder | undefined>();\n  const [selectedFolder, _setSelectedFolder] = useState<IFolder | 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 setSelectedFolder = (folder: IFolder) => {\n    _setSelectedFolder(folder);\n    onSelectFolder?.(folder.id);\n  };\n\n  const setFocusedFolder = (folder: IFolder) => {\n    _setFocusedFolder(folder);\n    setNewFolderParentId(undefined);\n\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const onOpenFolder = (id: string) => {\n    setOpenFolders(uniq(openFolders.concat(id)));\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 onNewFolderCreated = (newFolder: IFolder | undefined, parentId: string) => {\n    if (newFolder) {\n      setSelectedFolder(newFolder);\n      setFocusedFolder(newFolder);\n      setOpenFolders(uniq(openFolders.concat(parentId)));\n      setNewFolderParentId?.(undefined);\n      ref.current?.focus({ preventScroll: true });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolderParentId?.(undefined);\n    ref.current?.focus({ preventScroll: true });\n  };\n\n  const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maxLevel || 1);\n\n  return (\n    <StyledTreeStructure\n      onBlur={(e) => {\n        if (type === \"picker\" && !e.currentTarget.contains(e.relatedTarget)) {\n          onToggleTree(false);\n        }\n      }}\n    >\n      <Row>\n        {label && <StyledLabel id={treestructureId(type, \"label\")}>{label}</StyledLabel>}\n        {type === \"picker\" && (\n          <AddFolderButton\n            loading={loading}\n            canAddFolder={!!canAddFolder}\n            focusedFolder={focusedFolder}\n            setNewFolderParentId={setNewFolderParentId}\n            setShowTree={setShowTree}\n          />\n        )}\n      </Row>\n      <TreeStructureWrapper aria-label={label} data-type={type}>\n        {type === \"picker\" && (\n          <ComboboxButton\n            ref={ref}\n            showTree={showTree}\n            type={type}\n            label={label}\n            loading={loading}\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            ariaDescribedby={ariaDescribedby}\n          />\n        )}\n        {showTree && (\n          <ScrollableDiv data-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              openFolders={openFolders}\n              setFocusedFolder={setFocusedFolder}\n              setSelectedFolder={setSelectedFolder}\n              targetResource={targetResource}\n              visibleFolders={flattenedFolders}\n              type={type}\n              closeTree={() => onToggleTree(false)}\n              newFolderInput={newFolderInput}\n              onCreate={onNewFolderCreated}\n            />\n          </ScrollableDiv>\n        )}\n      </TreeStructureWrapper>\n    </StyledTreeStructure>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
61
80
  const TreeStructure = _ref => {
62
81
  let {
63
- defaultOpenFolders,
64
82
  folders,
83
+ defaultOpenFolders,
84
+ newFolderInput,
65
85
  label,
86
+ targetResource,
66
87
  loading,
67
88
  maxLevel = MAX_LEVEL_FOR_FOLDERS,
68
89
  onSelectFolder,
69
- targetResource,
70
- type,
71
- newFolderInput,
72
90
  ariaDescribedby
73
91
  } = _ref;
74
- const ref = (0, _react.useRef)(null);
75
- const defaultSelectedFolderId = defaultOpenFolders?.[defaultOpenFolders.length - 1];
76
- const [openFolders, setOpenFolders] = (0, _react.useState)(defaultOpenFolders || []);
77
- const [newFolderParentId, setNewFolderParentId] = (0, _react.useState)();
78
- const [focusedFolder, _setFocusedFolder] = (0, _react.useState)();
79
- const [selectedFolder, _setSelectedFolder] = (0, _react.useState)();
80
- const [showTree, setShowTree] = (0, _react.useState)(type === "navigation");
81
- const flattenedFolders = (0, _react.useMemo)(() => (0, _helperFunctions.flattenFolders)(folders, openFolders), [folders, openFolders]);
82
- (0, _react.useEffect)(() => {
83
- if (defaultOpenFolders) {
84
- if (!defaultOpenFolders.every(element => openFolders.includes(element))) {
85
- setOpenFolders(prev => {
86
- return uniq(defaultOpenFolders.concat(prev));
87
- });
88
- }
89
- }
90
- // eslint-disable-next-line react-hooks/exhaustive-deps
91
- }, [defaultOpenFolders]);
92
- (0, _react.useEffect)(() => {
93
- if (defaultSelectedFolderId !== undefined) {
94
- const selected = (0, _helperFunctions.flattenFolders)(folders).find(folder => folder.id === defaultSelectedFolderId);
95
- if (selected) {
96
- _setSelectedFolder(selected);
97
- if (type === "picker") {
98
- _setFocusedFolder(selected);
99
- }
100
- }
92
+ const [open, setOpen] = (0, _react.useState)(false);
93
+ const [selectedValue, setSelectedValue] = (0, _react.useState)(defaultOpenFolders?.[defaultOpenFolders?.length - 1] ?? "");
94
+ const [expandedValue, setExpandedValue] = (0, _react.useState)(defaultOpenFolders ?? []);
95
+ const [focusedValue, setFocusedValue] = (0, _react.useState)(selectedValue);
96
+ const [newFolderParentId, setNewFolderParentId] = (0, _react.useState)(null);
97
+ const newFolderButtonRef = (0, _react.useRef)(null);
98
+ const {
99
+ t
100
+ } = (0, _reactI18next.useTranslation)();
101
+ const rootFolderIds = (0, _react.useMemo)(() => folders.map(folder => folder.id), [folders]);
102
+ const contentRef = (0, _react.useRef)(null);
103
+ const selectedFolder = (0, _react.useMemo)(() => {
104
+ return (0, _helperFunctions.flattenFolders)(folders).find(folder => folder.id === selectedValue);
105
+ }, [folders, selectedValue]);
106
+ const disableCreateFolder = (0, _react.useMemo)(() => {
107
+ return (selectedFolder?.breadcrumbs.length ?? 0) > maxLevel - 1;
108
+ }, [maxLevel, selectedFolder?.breadcrumbs.length]);
109
+ const onOpenChange = (0, _react.useCallback)(details => {
110
+ setOpen(details.open);
111
+ if (!details.open) {
112
+ setNewFolderParentId(null);
101
113
  }
102
- // eslint-disable-next-line react-hooks/exhaustive-deps
103
- }, [defaultSelectedFolderId]);
104
- const onToggleTree = open => {
105
- setShowTree(open);
106
- if (!open) {
107
- setNewFolderParentId(undefined);
114
+ }, []);
115
+ const onKeyDown = (0, _react.useCallback)(e => {
116
+ if (e.key === "ArrowUp" || e.key === "ArrowDown") {
117
+ e.stopPropagation();
118
+ setOpen(true);
108
119
  }
109
- };
110
- const setSelectedFolder = folder => {
111
- _setSelectedFolder(folder);
112
- onSelectFolder?.(folder.id);
113
- };
114
- const setFocusedFolder = folder => {
115
- _setFocusedFolder(folder);
116
- setNewFolderParentId(undefined);
117
- ref.current?.focus({
118
- preventScroll: true
119
- });
120
- };
121
- const onOpenFolder = id => {
122
- setOpenFolders(uniq(openFolders.concat(id)));
123
- };
124
- const onCloseFolder = id => {
125
- const closedFolder = flattenedFolders.find(folder => folder.id === id);
126
- if (closedFolder) {
127
- const subFolders = closedFolder.subfolders && (0, _helperFunctions.flattenFolders)(closedFolder.subfolders);
128
- if (subFolders.some(folder => folder.id === selectedFolder?.id)) {
129
- setFocusedFolder(closedFolder);
120
+ }, []);
121
+ const onShowInput = (0, _react.useCallback)(() => {
122
+ if (disableCreateFolder) return;
123
+ const flattenedFolders = (0, _helperFunctions.flattenFolders)(folders);
124
+ const folder = flattenedFolders.find(folder => folder.id === selectedValue);
125
+ const newExpandedIds = rootFolderIds.concat(folder?.breadcrumbs.map(bc => bc.id) ?? []);
126
+ setOpen(true);
127
+ setExpandedValue(prev => Array.from(new Set([...prev, ...newExpandedIds])));
128
+ setNewFolderParentId(selectedValue);
129
+ }, [disableCreateFolder, folders, rootFolderIds, selectedValue]);
130
+ const treeView = (0, _react2.useTreeView)({
131
+ focusedValue,
132
+ onFocusChange: details => !!details.focusedValue && setFocusedValue(details.focusedValue),
133
+ expandedValue,
134
+ onExpandedChange: details => setExpandedValue(details.expandedValue),
135
+ selectedValue: [selectedValue],
136
+ onSelectionChange: details => {
137
+ // TODO: This is currently a bug in zag. The TreeView component simply expects the already selected value to remain selected. As such, always choose the "last" selected value.
138
+ const val = details.selectedValue[details.selectedValue.length - 1];
139
+ if (!val) return;
140
+ if (val === selectedValue && details.focusedValue === selectedValue) {
141
+ setOpen(false);
142
+ return;
130
143
  }
131
- }
132
- setOpenFolders(openFolders.filter(folderId => folderId !== id));
133
- };
134
- const onNewFolderCreated = (newFolder, parentId) => {
135
- if (newFolder) {
136
- setSelectedFolder(newFolder);
137
- setFocusedFolder(newFolder);
138
- setOpenFolders(uniq(openFolders.concat(parentId)));
139
- setNewFolderParentId?.(undefined);
140
- ref.current?.focus({
141
- preventScroll: true
144
+ setSelectedValue(val);
145
+ onSelectFolder?.(val);
146
+ },
147
+ expandOnClick: false
148
+ });
149
+ const onCreateFolder = (0, _react.useCallback)(folder => {
150
+ if (!folder) return;
151
+ const focus = treeView.focusItem;
152
+ const expand = treeView.expand;
153
+ const select = treeView.select;
154
+ (0, _reactDom.flushSync)(() => {
155
+ setOpen(true);
156
+ });
157
+ (0, _reactDom.flushSync)(() => {
158
+ expand(folder.breadcrumbs.map(bc => bc.id));
159
+ });
160
+ (0, _reactDom.flushSync)(() => {
161
+ select([folder.id]);
162
+ });
163
+ (0, _reactDom.flushSync)(() => {
164
+ focus(folder.id);
165
+ });
166
+ setNewFolderParentId(null);
167
+ }, [treeView.expand, treeView.focusItem, treeView.select]);
168
+ const onAnimationEnd = (0, _react.useCallback)(() => {
169
+ if (open && focusedValue) {
170
+ document.getElementById(focusedValue)?.scrollIntoView({
171
+ behavior: "smooth",
172
+ block: "nearest"
142
173
  });
143
174
  }
144
- };
145
- const onCancelNewFolder = () => {
146
- setNewFolderParentId?.(undefined);
147
- ref.current?.focus({
148
- preventScroll: true
149
- });
150
- };
151
- const canAddFolder = selectedFolder && selectedFolder?.breadcrumbs.length < (maxLevel || 1);
152
- return /*#__PURE__*/(0, _jsxRuntime.jsxs)(StyledTreeStructure, {
153
- onBlur: e => {
154
- if (type === "picker" && !e.currentTarget.contains(e.relatedTarget)) {
155
- onToggleTree(false);
156
- }
157
- },
158
- children: [/*#__PURE__*/(0, _jsxRuntime.jsxs)(Row, {
159
- children: [label && /*#__PURE__*/(0, _jsxRuntime.jsx)(StyledLabel, {
160
- id: (0, _helperFunctions.treestructureId)(type, "label"),
161
- children: label
162
- }), type === "picker" && /*#__PURE__*/(0, _jsxRuntime.jsx)(_AddFolderButton.default, {
163
- loading: loading,
164
- canAddFolder: !!canAddFolder,
165
- focusedFolder: focusedFolder,
166
- setNewFolderParentId: setNewFolderParentId,
167
- setShowTree: setShowTree
168
- })]
169
- }), /*#__PURE__*/(0, _jsxRuntime.jsxs)(TreeStructureWrapper, {
170
- "aria-label": label,
171
- "data-type": type,
172
- children: [type === "picker" && /*#__PURE__*/(0, _jsxRuntime.jsx)(_ComboboxButton.default, {
173
- ref: ref,
174
- showTree: showTree,
175
- type: type,
176
- label: label,
177
- loading: loading,
178
- focusedFolder: focusedFolder,
179
- selectedFolder: selectedFolder,
180
- setSelectedFolder: setSelectedFolder,
181
- setFocusedFolder: setFocusedFolder,
182
- onToggleTree: onToggleTree,
183
- flattenedFolders: flattenedFolders,
184
- onCloseFolder: onCloseFolder,
185
- onOpenFolder: onOpenFolder,
186
- ariaDescribedby: ariaDescribedby
187
- }), showTree && /*#__PURE__*/(0, _jsxRuntime.jsx)(ScrollableDiv, {
188
- "data-type": type,
189
- children: /*#__PURE__*/(0, _jsxRuntime.jsx)(_FolderItems.default, {
190
- focusedFolder: focusedFolder,
191
- folders: folders,
192
- level: 0,
175
+ }, [focusedValue, open]);
176
+ const onCancelFolder = (0, _react.useCallback)(() => {
177
+ if (!selectedFolder) return;
178
+ const focusFunc = selectedFolder.subfolders.length ? treeView.focusBranch : treeView.focusItem;
179
+ focusFunc(selectedFolder.id);
180
+ setNewFolderParentId(null);
181
+ }, [selectedFolder, treeView.focusBranch, treeView.focusItem]);
182
+ const addTooltip = loading ? t("loading") : disableCreateFolder ? t("treeStructure.maxFoldersAlreadyAdded") : t("myNdla.newFolderUnder", {
183
+ folderName: selectedFolder?.name
184
+ });
185
+ return /*#__PURE__*/(0, _jsxRuntime.jsx)(StyledTreeRootProvider, {
186
+ value: treeView,
187
+ asChild: true,
188
+ ...treeView.getRootProps(),
189
+ children: /*#__PURE__*/(0, _jsxRuntime.jsxs)(_jsx2.Stack, {
190
+ align: "flex-end",
191
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsxs)(LabelHStack, {
192
+ gap: "xsmall",
193
+ justify: "space-between",
194
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeLabel, {
195
+ children: label
196
+ }), /*#__PURE__*/(0, _jsxRuntime.jsxs)(_primitives.Button, {
197
+ size: "small",
198
+ variant: "tertiary",
199
+ ref: newFolderButtonRef,
200
+ "aria-disabled": disableCreateFolder,
201
+ title: addTooltip,
202
+ "aria-label": addTooltip,
193
203
  loading: loading,
194
- selectedFolder: selectedFolder,
195
- maxLevel: maxLevel,
196
- newFolderParentId: newFolderParentId,
197
- onCancelNewFolder: onCancelNewFolder,
198
- onCloseFolder: onCloseFolder,
199
- onOpenFolder: onOpenFolder,
200
- openFolders: openFolders,
201
- setFocusedFolder: setFocusedFolder,
202
- setSelectedFolder: setSelectedFolder,
203
- targetResource: targetResource,
204
- visibleFolders: flattenedFolders,
205
- type: type,
206
- closeTree: () => onToggleTree(false),
207
- newFolderInput: newFolderInput,
208
- onCreate: onNewFolderCreated
209
- })
204
+ onClick: onShowInput,
205
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(_action.AddLine, {}), t("myNdla.newFolder")]
206
+ })]
207
+ }), /*#__PURE__*/(0, _jsxRuntime.jsxs)(_primitives.PopoverRoot, {
208
+ open: open,
209
+ positioning: {
210
+ sameWidth: true
211
+ },
212
+ onOpenChange: onOpenChange,
213
+ persistentElements: [() => newFolderButtonRef.current],
214
+ initialFocusEl: () => contentRef.current?.querySelector("input") ?? null,
215
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.PopoverTrigger, {
216
+ asChild: true,
217
+ children: /*#__PURE__*/(0, _jsxRuntime.jsxs)(StyledButton, {
218
+ variant: "secondary",
219
+ onKeyDown: onKeyDown,
220
+ "aria-haspopup": "tree",
221
+ role: "combobox",
222
+ "aria-describedby": ariaDescribedby,
223
+ "aria-activedescendant": focusedValue ?? undefined,
224
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)("span", {
225
+ children: selectedFolder?.name
226
+ }), /*#__PURE__*/(0, _jsxRuntime.jsx)(_common.ArrowDownShortLine, {})]
227
+ })
228
+ }), /*#__PURE__*/(0, _jsxRuntime.jsxs)(StyledPopoverContent, {
229
+ onAnimationEnd: onAnimationEnd,
230
+ ref: contentRef,
231
+ children: [!!newFolderParentId && newFolderInput?.({
232
+ parentId: newFolderParentId,
233
+ onCreate: onCreateFolder,
234
+ onCancel: onCancelFolder
235
+ }), /*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.Tree, {
236
+ children: folders.map(folder => /*#__PURE__*/(0, _jsxRuntime.jsx)(TreeStructureItem, {
237
+ folder: folder,
238
+ targetResource: targetResource
239
+ }, folder.id))
240
+ })]
241
+ })]
210
242
  })]
211
- })]
243
+ })
212
244
  });
213
245
  };
214
- var _default = exports.default = TreeStructure;
246
+ exports.TreeStructure = TreeStructure;
247
+ const TreeStructureItem = _ref2 => {
248
+ let {
249
+ folder,
250
+ targetResource
251
+ } = _ref2;
252
+ const {
253
+ t
254
+ } = (0, _reactI18next.useTranslation)();
255
+ const {
256
+ setOpen
257
+ } = (0, _react2.usePopoverContext)();
258
+ const containsResource = targetResource && folder.resources.some(resource => resource.resourceId === targetResource.resourceId);
259
+ const FolderIcon = folder.status === "shared" ? StyledFolderUserLine : StyledFolderLine;
260
+
261
+ // TODO: Pressing enter selects the item and closes the popover immediately. Do we actually want this? Old behavior.
262
+ const onKeyDown = (0, _react.useCallback)(e => {
263
+ if (e.key === "Enter") {
264
+ setOpen(false);
265
+ }
266
+ }, [setOpen]);
267
+ if (!folder.subfolders.length) {
268
+ return /*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeItem, {
269
+ value: folder.id,
270
+ onKeyDown: onKeyDown,
271
+ id: folder.id,
272
+ children: /*#__PURE__*/(0, _jsxRuntime.jsxs)(StyledHStack, {
273
+ gap: "xsmall",
274
+ justify: "space-between",
275
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsxs)(StyledHStack, {
276
+ gap: "xxsmall",
277
+ justify: "center",
278
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(FolderIcon, {}), /*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeItemText, {
279
+ children: folder.name
280
+ })]
281
+ }), containsResource && /*#__PURE__*/(0, _jsxRuntime.jsx)(StyledHeartFill, {
282
+ title: t("myNdla.alreadyInFolder")
283
+ })]
284
+ })
285
+ }, folder.id);
286
+ }
287
+ const ariaLabel = folder.status === "shared" ? `${folder.name}. ${t("myNdla.folder.sharing.shared")}` : folder.name;
288
+ return /*#__PURE__*/(0, _jsxRuntime.jsxs)(_primitives.TreeBranch, {
289
+ value: folder.id,
290
+ id: folder.id,
291
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeBranchControl, {
292
+ onKeyDown: onKeyDown,
293
+ asChild: true,
294
+ children: /*#__PURE__*/(0, _jsxRuntime.jsxs)(StyledHStack, {
295
+ gap: "xsmall",
296
+ justify: "space-between",
297
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsxs)(StyledHStack, {
298
+ gap: "xxsmall",
299
+ justify: "center",
300
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.IconButton, {
301
+ variant: "clear",
302
+ asChild: true,
303
+ children: /*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeBranchTrigger, {
304
+ children: /*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeBranchIndicator, {
305
+ asChild: true,
306
+ children: /*#__PURE__*/(0, _jsxRuntime.jsx)(_common.ArrowRightShortLine, {})
307
+ })
308
+ })
309
+ }), /*#__PURE__*/(0, _jsxRuntime.jsx)(FolderIcon, {}), /*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeBranchText, {
310
+ "aria-label": ariaLabel,
311
+ title: ariaLabel,
312
+ children: folder.name
313
+ })]
314
+ }), containsResource && /*#__PURE__*/(0, _jsxRuntime.jsx)(StyledHeartFill, {
315
+ title: t("myNdla.alreadyInFolder"),
316
+ "aria-label": t("myNdla.alreadyInFolder"),
317
+ "aria-hidden": false
318
+ })]
319
+ })
320
+ }), /*#__PURE__*/(0, _jsxRuntime.jsx)(_primitives.TreeBranchContent, {
321
+ children: folder.subfolders.map(subfolder => /*#__PURE__*/(0, _jsxRuntime.jsx)(TreeStructureItem, {
322
+ folder: subfolder,
323
+ targetResource: targetResource
324
+ }, subfolder.id))
325
+ })]
326
+ }, folder.id);
327
+ };
@@ -6,6 +6,4 @@
6
6
  *
7
7
  */
8
8
  import { IFolder } from "@ndla/types-backend/myndla-api";
9
- import { TreeStructureType } from "./types";
10
9
  export declare const flattenFolders: (folders: IFolder[], openFolders?: string[]) => IFolder[];
11
- export declare const treestructureId: (type: TreeStructureType, modifier: string) => string;