@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.
- package/dist/panda.buildinfo.json +7 -2
- package/dist/styles.css +24 -4
- package/es/Article/Article.js +4 -1
- package/es/ContentTypeBadge/ContentTypeBadge.js +2 -0
- package/es/TreeStructure/TreeStructure.js +292 -181
- package/es/TreeStructure/helperFunctions.js +0 -3
- package/es/TreeStructure/index.js +1 -2
- package/es/index.js +0 -1
- package/es/locale/messages-en.js +1 -1
- package/es/locale/messages-nb.js +1 -1
- package/es/locale/messages-nn.js +1 -1
- package/es/locale/messages-se.js +1 -1
- package/es/locale/messages-sma.js +1 -1
- package/es/styles.css +24 -4
- package/lib/Article/Article.js +4 -1
- package/lib/ContentTypeBadge/ContentTypeBadge.js +2 -0
- package/lib/TreeStructure/TreeStructure.d.ts +7 -6
- package/lib/TreeStructure/TreeStructure.js +293 -180
- package/lib/TreeStructure/helperFunctions.d.ts +0 -2
- package/lib/TreeStructure/helperFunctions.js +2 -6
- package/lib/TreeStructure/index.d.ts +1 -2
- package/lib/TreeStructure/index.js +2 -3
- package/lib/TreeStructure/types.d.ts +4 -22
- package/lib/index.d.ts +0 -2
- package/lib/index.js +0 -7
- package/lib/locale/messages-en.js +1 -1
- package/lib/locale/messages-nb.js +1 -1
- package/lib/locale/messages-nn.js +1 -1
- package/lib/locale/messages-se.js +1 -1
- package/lib/locale/messages-sma.js +1 -1
- package/lib/styles.css +24 -4
- package/package.json +7 -8
- package/src/Article/Article.tsx +4 -1
- package/src/ContentTypeBadge/ContentTypeBadge.tsx +2 -0
- package/src/TreeStructure/TreeStructure.stories.tsx +38 -68
- package/src/TreeStructure/TreeStructure.tsx +307 -194
- package/src/TreeStructure/helperFunctions.ts +0 -5
- package/src/TreeStructure/index.ts +1 -2
- package/src/TreeStructure/types.ts +4 -25
- package/src/index.ts +0 -3
- package/src/locale/messages-en.ts +1 -1
- package/src/locale/messages-nb.ts +1 -1
- package/src/locale/messages-nn.ts +1 -1
- package/src/locale/messages-se.ts +1 -1
- package/src/locale/messages-sma.ts +1 -1
- package/es/ProgrammeCard/ProgrammeCard.js +0 -51
- package/es/ProgrammeCard/index.js +0 -9
- package/es/TreeStructure/AddFolderButton.js +0 -80
- package/es/TreeStructure/ComboboxButton.js +0 -127
- package/es/TreeStructure/FolderItem.js +0 -225
- package/es/TreeStructure/FolderItems.js +0 -95
- package/es/TreeStructure/arrowNavigation.js +0 -35
- package/lib/ProgrammeCard/ProgrammeCard.d.ts +0 -23
- package/lib/ProgrammeCard/ProgrammeCard.js +0 -58
- package/lib/ProgrammeCard/index.d.ts +0 -9
- package/lib/ProgrammeCard/index.js +0 -13
- package/lib/TreeStructure/AddFolderButton.d.ts +0 -17
- package/lib/TreeStructure/AddFolderButton.js +0 -85
- package/lib/TreeStructure/ComboboxButton.d.ts +0 -27
- package/lib/TreeStructure/ComboboxButton.js +0 -134
- package/lib/TreeStructure/FolderItem.d.ts +0 -17
- package/lib/TreeStructure/FolderItem.js +0 -230
- package/lib/TreeStructure/FolderItems.d.ts +0 -22
- package/lib/TreeStructure/FolderItems.js +0 -100
- package/lib/TreeStructure/arrowNavigation.d.ts +0 -10
- package/lib/TreeStructure/arrowNavigation.js +0 -42
- package/src/ProgrammeCard/ProgrammeCard.stories.tsx +0 -35
- package/src/ProgrammeCard/ProgrammeCard.tsx +0 -78
- package/src/ProgrammeCard/index.tsx +0 -10
- package/src/TreeStructure/AddFolderButton.tsx +0 -79
- package/src/TreeStructure/ComboboxButton.tsx +0 -172
- package/src/TreeStructure/FolderItem.tsx +0 -307
- package/src/TreeStructure/FolderItems.tsx +0 -121
- package/src/TreeStructure/arrowNavigation.ts +0 -54
|
@@ -1,209 +1,320 @@
|
|
|
1
|
-
import _styled from "@emotion/styled/base";
|
|
2
|
-
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)."; }
|
|
3
1
|
/**
|
|
4
|
-
* Copyright (c)
|
|
2
|
+
* Copyright (c) 2024-present, NDLA.
|
|
5
3
|
*
|
|
6
4
|
* This source code is licensed under the GPLv3 license found in the
|
|
7
5
|
* LICENSE file in the root directory of this source tree.
|
|
8
6
|
*
|
|
9
7
|
*/
|
|
10
8
|
|
|
11
|
-
import {
|
|
12
|
-
import {
|
|
13
|
-
import
|
|
14
|
-
import
|
|
15
|
-
import
|
|
16
|
-
import {
|
|
9
|
+
import { useCallback, useMemo, useRef, useState } from "react";
|
|
10
|
+
import { flushSync } from "react-dom";
|
|
11
|
+
import { useTranslation } from "react-i18next";
|
|
12
|
+
import { useTreeView, usePopoverContext } from "@ark-ui/react";
|
|
13
|
+
import { AddLine, HeartFill } from "@ndla/icons/action";
|
|
14
|
+
import { ArrowDownShortLine, ArrowRightShortLine } from "@ndla/icons/common";
|
|
15
|
+
import { FolderUserLine } from "@ndla/icons/contentType";
|
|
16
|
+
import { FolderLine } from "@ndla/icons/editor";
|
|
17
|
+
import { Button, IconButton, PopoverContent, PopoverRoot, PopoverTrigger, Tree, TreeBranch, TreeBranchContent, TreeBranchControl, TreeBranchIndicator, TreeBranchText, TreeBranchTrigger, TreeItem, TreeItemText, TreeLabel, TreeRootProvider } from "@ndla/primitives";
|
|
18
|
+
import { HStack, Stack, styled } from "@ndla/styled-system/jsx";
|
|
19
|
+
import { flattenFolders } from "./helperFunctions";
|
|
17
20
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
18
21
|
export const MAX_LEVEL_FOR_FOLDERS = 5;
|
|
19
|
-
const
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
})("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":"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"]} */"));
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
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"]} */",
|
|
34
|
-
|
|
22
|
+
const StyledButton = styled(Button, {
|
|
23
|
+
base: {
|
|
24
|
+
width: "100%",
|
|
25
|
+
justifyContent: "space-between",
|
|
26
|
+
"& span": {
|
|
27
|
+
overflow: "hidden",
|
|
28
|
+
textOverflow: "ellipsis"
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
});
|
|
32
|
+
const StyledHStack = styled(HStack, {
|
|
33
|
+
base: {
|
|
34
|
+
overflow: "hidden"
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
const StyledHeartFill = styled(HeartFill, {
|
|
38
|
+
base: {
|
|
39
|
+
color: "icon.strong"
|
|
40
|
+
}
|
|
41
|
+
});
|
|
42
|
+
const StyledFolderLine = styled(FolderLine, {
|
|
43
|
+
base: {
|
|
44
|
+
color: "icon.strong"
|
|
45
|
+
}
|
|
46
|
+
});
|
|
47
|
+
const StyledFolderUserLine = styled(FolderUserLine, {
|
|
48
|
+
base: {
|
|
49
|
+
color: "icon.strong"
|
|
50
|
+
}
|
|
35
51
|
});
|
|
36
|
-
const
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
styles: "display:flex;align-items:center;justify-content:space-between"
|
|
42
|
-
} : {
|
|
43
|
-
name: "bcffy2",
|
|
44
|
-
styles: "display:flex;align-items:center;justify-content:space-between",
|
|
45
|
-
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"]} */",
|
|
46
|
-
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
52
|
+
const StyledTreeRootProvider = styled(TreeRootProvider, {
|
|
53
|
+
base: {
|
|
54
|
+
width: "100%",
|
|
55
|
+
maxHeight: "inherit"
|
|
56
|
+
}
|
|
47
57
|
});
|
|
48
|
-
const
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
})("display:flex;flex-direction:column;transition:", misc.transition.default, ";&[data-type=\"picker\"]{overflow:hidden;border:1px solid ", colors.brand.neutral7, ";border-radius:", misc.borderRadius, ";scroll-behavior:smooth;}&: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":"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"]} */"));
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
})("&[data-type=\"picker\"]{overflow:auto;overflow:overlay;", 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"]} */"));
|
|
56
|
-
|
|
58
|
+
const StyledPopoverContent = styled(PopoverContent, {
|
|
59
|
+
base: {
|
|
60
|
+
display: "flex",
|
|
61
|
+
flexDirection: "column",
|
|
62
|
+
gap: "xsmall",
|
|
63
|
+
overflow: "auto",
|
|
64
|
+
maxHeight: "inherit",
|
|
65
|
+
paddingInline: "xsmall",
|
|
66
|
+
paddingBlock: "xsmall"
|
|
67
|
+
}
|
|
68
|
+
});
|
|
69
|
+
const LabelHStack = styled(HStack, {
|
|
70
|
+
base: {
|
|
71
|
+
width: "100%"
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
export const TreeStructure = _ref => {
|
|
57
75
|
let {
|
|
58
|
-
defaultOpenFolders,
|
|
59
76
|
folders,
|
|
77
|
+
defaultOpenFolders,
|
|
78
|
+
newFolderInput,
|
|
60
79
|
label,
|
|
80
|
+
targetResource,
|
|
61
81
|
loading,
|
|
62
82
|
maxLevel = MAX_LEVEL_FOR_FOLDERS,
|
|
63
83
|
onSelectFolder,
|
|
64
|
-
targetResource,
|
|
65
|
-
type,
|
|
66
|
-
newFolderInput,
|
|
67
84
|
ariaDescribedby
|
|
68
85
|
} = _ref;
|
|
69
|
-
const
|
|
70
|
-
const
|
|
71
|
-
const [
|
|
72
|
-
const [
|
|
73
|
-
const [
|
|
74
|
-
const
|
|
75
|
-
const
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
if (
|
|
89
|
-
|
|
90
|
-
if (selected) {
|
|
91
|
-
_setSelectedFolder(selected);
|
|
92
|
-
if (type === "picker") {
|
|
93
|
-
_setFocusedFolder(selected);
|
|
94
|
-
}
|
|
95
|
-
}
|
|
86
|
+
const [open, setOpen] = useState(false);
|
|
87
|
+
const [selectedValue, setSelectedValue] = useState(defaultOpenFolders?.[defaultOpenFolders?.length - 1] ?? "");
|
|
88
|
+
const [expandedValue, setExpandedValue] = useState(defaultOpenFolders ?? []);
|
|
89
|
+
const [focusedValue, setFocusedValue] = useState(selectedValue);
|
|
90
|
+
const [newFolderParentId, setNewFolderParentId] = useState(null);
|
|
91
|
+
const newFolderButtonRef = useRef(null);
|
|
92
|
+
const {
|
|
93
|
+
t
|
|
94
|
+
} = useTranslation();
|
|
95
|
+
const rootFolderIds = useMemo(() => folders.map(folder => folder.id), [folders]);
|
|
96
|
+
const contentRef = useRef(null);
|
|
97
|
+
const selectedFolder = useMemo(() => {
|
|
98
|
+
return flattenFolders(folders).find(folder => folder.id === selectedValue);
|
|
99
|
+
}, [folders, selectedValue]);
|
|
100
|
+
const disableCreateFolder = useMemo(() => {
|
|
101
|
+
return (selectedFolder?.breadcrumbs.length ?? 0) > maxLevel - 1;
|
|
102
|
+
}, [maxLevel, selectedFolder?.breadcrumbs.length]);
|
|
103
|
+
const onOpenChange = useCallback(details => {
|
|
104
|
+
setOpen(details.open);
|
|
105
|
+
if (!details.open) {
|
|
106
|
+
setNewFolderParentId(null);
|
|
96
107
|
}
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
setNewFolderParentId(undefined);
|
|
108
|
+
}, []);
|
|
109
|
+
const onKeyDown = useCallback(e => {
|
|
110
|
+
if (e.key === "ArrowUp" || e.key === "ArrowDown") {
|
|
111
|
+
e.stopPropagation();
|
|
112
|
+
setOpen(true);
|
|
103
113
|
}
|
|
104
|
-
};
|
|
105
|
-
const
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
const
|
|
123
|
-
if (
|
|
124
|
-
|
|
114
|
+
}, []);
|
|
115
|
+
const onShowInput = useCallback(() => {
|
|
116
|
+
if (disableCreateFolder) return;
|
|
117
|
+
const flattenedFolders = flattenFolders(folders);
|
|
118
|
+
const folder = flattenedFolders.find(folder => folder.id === selectedValue);
|
|
119
|
+
const newExpandedIds = rootFolderIds.concat(folder?.breadcrumbs.map(bc => bc.id) ?? []);
|
|
120
|
+
setOpen(true);
|
|
121
|
+
setExpandedValue(prev => Array.from(new Set([...prev, ...newExpandedIds])));
|
|
122
|
+
setNewFolderParentId(selectedValue);
|
|
123
|
+
}, [disableCreateFolder, folders, rootFolderIds, selectedValue]);
|
|
124
|
+
const treeView = useTreeView({
|
|
125
|
+
focusedValue,
|
|
126
|
+
onFocusChange: details => !!details.focusedValue && setFocusedValue(details.focusedValue),
|
|
127
|
+
expandedValue,
|
|
128
|
+
onExpandedChange: details => setExpandedValue(details.expandedValue),
|
|
129
|
+
selectedValue: [selectedValue],
|
|
130
|
+
onSelectionChange: details => {
|
|
131
|
+
// 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.
|
|
132
|
+
const val = details.selectedValue[details.selectedValue.length - 1];
|
|
133
|
+
if (!val) return;
|
|
134
|
+
if (val === selectedValue && details.focusedValue === selectedValue) {
|
|
135
|
+
setOpen(false);
|
|
136
|
+
return;
|
|
125
137
|
}
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
138
|
+
setSelectedValue(val);
|
|
139
|
+
onSelectFolder?.(val);
|
|
140
|
+
},
|
|
141
|
+
expandOnClick: false
|
|
142
|
+
});
|
|
143
|
+
const onCreateFolder = useCallback(folder => {
|
|
144
|
+
if (!folder) return;
|
|
145
|
+
const focus = treeView.focusItem;
|
|
146
|
+
const expand = treeView.expand;
|
|
147
|
+
const select = treeView.select;
|
|
148
|
+
flushSync(() => {
|
|
149
|
+
setOpen(true);
|
|
150
|
+
});
|
|
151
|
+
flushSync(() => {
|
|
152
|
+
expand(folder.breadcrumbs.map(bc => bc.id));
|
|
153
|
+
});
|
|
154
|
+
flushSync(() => {
|
|
155
|
+
select([folder.id]);
|
|
156
|
+
});
|
|
157
|
+
flushSync(() => {
|
|
158
|
+
focus(folder.id);
|
|
159
|
+
});
|
|
160
|
+
setNewFolderParentId(null);
|
|
161
|
+
}, [treeView.expand, treeView.focusItem, treeView.select]);
|
|
162
|
+
const onAnimationEnd = useCallback(() => {
|
|
163
|
+
if (open && focusedValue) {
|
|
164
|
+
document.getElementById(focusedValue)?.scrollIntoView({
|
|
165
|
+
behavior: "smooth",
|
|
166
|
+
block: "nearest"
|
|
137
167
|
});
|
|
138
168
|
}
|
|
139
|
-
};
|
|
140
|
-
const
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
};
|
|
146
|
-
const
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
children:
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
children: [type === "picker" && /*#__PURE__*/_jsx(ComboboxButton, {
|
|
168
|
-
ref: ref,
|
|
169
|
-
showTree: showTree,
|
|
170
|
-
type: type,
|
|
171
|
-
label: label,
|
|
172
|
-
loading: loading,
|
|
173
|
-
focusedFolder: focusedFolder,
|
|
174
|
-
selectedFolder: selectedFolder,
|
|
175
|
-
setSelectedFolder: setSelectedFolder,
|
|
176
|
-
setFocusedFolder: setFocusedFolder,
|
|
177
|
-
onToggleTree: onToggleTree,
|
|
178
|
-
flattenedFolders: flattenedFolders,
|
|
179
|
-
onCloseFolder: onCloseFolder,
|
|
180
|
-
onOpenFolder: onOpenFolder,
|
|
181
|
-
ariaDescribedby: ariaDescribedby
|
|
182
|
-
}), showTree && /*#__PURE__*/_jsx(ScrollableDiv, {
|
|
183
|
-
"data-type": type,
|
|
184
|
-
children: /*#__PURE__*/_jsx(FolderItems, {
|
|
185
|
-
focusedFolder: focusedFolder,
|
|
186
|
-
folders: folders,
|
|
187
|
-
level: 0,
|
|
169
|
+
}, [focusedValue, open]);
|
|
170
|
+
const onCancelFolder = useCallback(() => {
|
|
171
|
+
if (!selectedFolder) return;
|
|
172
|
+
const focusFunc = selectedFolder.subfolders.length ? treeView.focusBranch : treeView.focusItem;
|
|
173
|
+
focusFunc(selectedFolder.id);
|
|
174
|
+
setNewFolderParentId(null);
|
|
175
|
+
}, [selectedFolder, treeView.focusBranch, treeView.focusItem]);
|
|
176
|
+
const addTooltip = loading ? t("loading") : disableCreateFolder ? t("treeStructure.maxFoldersAlreadyAdded") : t("myNdla.newFolderUnder", {
|
|
177
|
+
folderName: selectedFolder?.name
|
|
178
|
+
});
|
|
179
|
+
return /*#__PURE__*/_jsx(StyledTreeRootProvider, {
|
|
180
|
+
value: treeView,
|
|
181
|
+
asChild: true,
|
|
182
|
+
...treeView.getRootProps(),
|
|
183
|
+
children: /*#__PURE__*/_jsxs(Stack, {
|
|
184
|
+
align: "flex-end",
|
|
185
|
+
children: [/*#__PURE__*/_jsxs(LabelHStack, {
|
|
186
|
+
gap: "xsmall",
|
|
187
|
+
justify: "space-between",
|
|
188
|
+
children: [/*#__PURE__*/_jsx(TreeLabel, {
|
|
189
|
+
children: label
|
|
190
|
+
}), /*#__PURE__*/_jsxs(Button, {
|
|
191
|
+
size: "small",
|
|
192
|
+
variant: "tertiary",
|
|
193
|
+
ref: newFolderButtonRef,
|
|
194
|
+
"aria-disabled": disableCreateFolder,
|
|
195
|
+
title: addTooltip,
|
|
196
|
+
"aria-label": addTooltip,
|
|
188
197
|
loading: loading,
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
198
|
+
onClick: onShowInput,
|
|
199
|
+
children: [/*#__PURE__*/_jsx(AddLine, {}), t("myNdla.newFolder")]
|
|
200
|
+
})]
|
|
201
|
+
}), /*#__PURE__*/_jsxs(PopoverRoot, {
|
|
202
|
+
open: open,
|
|
203
|
+
positioning: {
|
|
204
|
+
sameWidth: true
|
|
205
|
+
},
|
|
206
|
+
onOpenChange: onOpenChange,
|
|
207
|
+
persistentElements: [() => newFolderButtonRef.current],
|
|
208
|
+
initialFocusEl: () => contentRef.current?.querySelector("input") ?? null,
|
|
209
|
+
children: [/*#__PURE__*/_jsx(PopoverTrigger, {
|
|
210
|
+
asChild: true,
|
|
211
|
+
children: /*#__PURE__*/_jsxs(StyledButton, {
|
|
212
|
+
variant: "secondary",
|
|
213
|
+
onKeyDown: onKeyDown,
|
|
214
|
+
"aria-haspopup": "tree",
|
|
215
|
+
role: "combobox",
|
|
216
|
+
"aria-describedby": ariaDescribedby,
|
|
217
|
+
"aria-activedescendant": focusedValue ?? undefined,
|
|
218
|
+
children: [/*#__PURE__*/_jsx("span", {
|
|
219
|
+
children: selectedFolder?.name
|
|
220
|
+
}), /*#__PURE__*/_jsx(ArrowDownShortLine, {})]
|
|
221
|
+
})
|
|
222
|
+
}), /*#__PURE__*/_jsxs(StyledPopoverContent, {
|
|
223
|
+
onAnimationEnd: onAnimationEnd,
|
|
224
|
+
ref: contentRef,
|
|
225
|
+
children: [!!newFolderParentId && newFolderInput?.({
|
|
226
|
+
parentId: newFolderParentId,
|
|
227
|
+
onCreate: onCreateFolder,
|
|
228
|
+
onCancel: onCancelFolder
|
|
229
|
+
}), /*#__PURE__*/_jsx(Tree, {
|
|
230
|
+
children: folders.map(folder => /*#__PURE__*/_jsx(TreeStructureItem, {
|
|
231
|
+
folder: folder,
|
|
232
|
+
targetResource: targetResource
|
|
233
|
+
}, folder.id))
|
|
234
|
+
})]
|
|
235
|
+
})]
|
|
205
236
|
})]
|
|
206
|
-
})
|
|
237
|
+
})
|
|
207
238
|
});
|
|
208
239
|
};
|
|
209
|
-
|
|
240
|
+
const TreeStructureItem = _ref2 => {
|
|
241
|
+
let {
|
|
242
|
+
folder,
|
|
243
|
+
targetResource
|
|
244
|
+
} = _ref2;
|
|
245
|
+
const {
|
|
246
|
+
t
|
|
247
|
+
} = useTranslation();
|
|
248
|
+
const {
|
|
249
|
+
setOpen
|
|
250
|
+
} = usePopoverContext();
|
|
251
|
+
const containsResource = targetResource && folder.resources.some(resource => resource.resourceId === targetResource.resourceId);
|
|
252
|
+
const FolderIcon = folder.status === "shared" ? StyledFolderUserLine : StyledFolderLine;
|
|
253
|
+
|
|
254
|
+
// TODO: Pressing enter selects the item and closes the popover immediately. Do we actually want this? Old behavior.
|
|
255
|
+
const onKeyDown = useCallback(e => {
|
|
256
|
+
if (e.key === "Enter") {
|
|
257
|
+
setOpen(false);
|
|
258
|
+
}
|
|
259
|
+
}, [setOpen]);
|
|
260
|
+
if (!folder.subfolders.length) {
|
|
261
|
+
return /*#__PURE__*/_jsx(TreeItem, {
|
|
262
|
+
value: folder.id,
|
|
263
|
+
onKeyDown: onKeyDown,
|
|
264
|
+
id: folder.id,
|
|
265
|
+
children: /*#__PURE__*/_jsxs(StyledHStack, {
|
|
266
|
+
gap: "xsmall",
|
|
267
|
+
justify: "space-between",
|
|
268
|
+
children: [/*#__PURE__*/_jsxs(StyledHStack, {
|
|
269
|
+
gap: "xxsmall",
|
|
270
|
+
justify: "center",
|
|
271
|
+
children: [/*#__PURE__*/_jsx(FolderIcon, {}), /*#__PURE__*/_jsx(TreeItemText, {
|
|
272
|
+
children: folder.name
|
|
273
|
+
})]
|
|
274
|
+
}), containsResource && /*#__PURE__*/_jsx(StyledHeartFill, {
|
|
275
|
+
title: t("myNdla.alreadyInFolder")
|
|
276
|
+
})]
|
|
277
|
+
})
|
|
278
|
+
}, folder.id);
|
|
279
|
+
}
|
|
280
|
+
const ariaLabel = folder.status === "shared" ? `${folder.name}. ${t("myNdla.folder.sharing.shared")}` : folder.name;
|
|
281
|
+
return /*#__PURE__*/_jsxs(TreeBranch, {
|
|
282
|
+
value: folder.id,
|
|
283
|
+
id: folder.id,
|
|
284
|
+
children: [/*#__PURE__*/_jsx(TreeBranchControl, {
|
|
285
|
+
onKeyDown: onKeyDown,
|
|
286
|
+
asChild: true,
|
|
287
|
+
children: /*#__PURE__*/_jsxs(StyledHStack, {
|
|
288
|
+
gap: "xsmall",
|
|
289
|
+
justify: "space-between",
|
|
290
|
+
children: [/*#__PURE__*/_jsxs(StyledHStack, {
|
|
291
|
+
gap: "xxsmall",
|
|
292
|
+
justify: "center",
|
|
293
|
+
children: [/*#__PURE__*/_jsx(IconButton, {
|
|
294
|
+
variant: "clear",
|
|
295
|
+
asChild: true,
|
|
296
|
+
children: /*#__PURE__*/_jsx(TreeBranchTrigger, {
|
|
297
|
+
children: /*#__PURE__*/_jsx(TreeBranchIndicator, {
|
|
298
|
+
asChild: true,
|
|
299
|
+
children: /*#__PURE__*/_jsx(ArrowRightShortLine, {})
|
|
300
|
+
})
|
|
301
|
+
})
|
|
302
|
+
}), /*#__PURE__*/_jsx(FolderIcon, {}), /*#__PURE__*/_jsx(TreeBranchText, {
|
|
303
|
+
"aria-label": ariaLabel,
|
|
304
|
+
title: ariaLabel,
|
|
305
|
+
children: folder.name
|
|
306
|
+
})]
|
|
307
|
+
}), containsResource && /*#__PURE__*/_jsx(StyledHeartFill, {
|
|
308
|
+
title: t("myNdla.alreadyInFolder"),
|
|
309
|
+
"aria-label": t("myNdla.alreadyInFolder"),
|
|
310
|
+
"aria-hidden": false
|
|
311
|
+
})]
|
|
312
|
+
})
|
|
313
|
+
}), /*#__PURE__*/_jsx(TreeBranchContent, {
|
|
314
|
+
children: folder.subfolders.map(subfolder => /*#__PURE__*/_jsx(TreeStructureItem, {
|
|
315
|
+
folder: subfolder,
|
|
316
|
+
targetResource: targetResource
|
|
317
|
+
}, subfolder.id))
|
|
318
|
+
})]
|
|
319
|
+
}, folder.id);
|
|
320
|
+
};
|