@ndla/ui 18.0.1 → 19.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (47) hide show
  1. package/es/MyNdla/Resource/Folder.js +10 -9
  2. package/es/Resource/BlockResource.js +14 -8
  3. package/es/Resource/ListResource.js +15 -9
  4. package/es/Resource/resourceComponents.js +12 -11
  5. package/es/TreeStructure/FolderItems.js +7 -6
  6. package/es/TreeStructure/TreeStructure.js +16 -9
  7. package/es/locale/messages-en.js +1 -0
  8. package/es/locale/messages-nb.js +1 -0
  9. package/es/locale/messages-nn.js +1 -0
  10. package/es/locale/messages-se.js +1 -0
  11. package/es/locale/messages-sma.js +1 -0
  12. package/lib/MyNdla/Resource/Folder.d.ts +4 -3
  13. package/lib/MyNdla/Resource/Folder.js +10 -9
  14. package/lib/Resource/BlockResource.d.ts +4 -3
  15. package/lib/Resource/BlockResource.js +14 -7
  16. package/lib/Resource/ListResource.d.ts +4 -3
  17. package/lib/Resource/ListResource.js +15 -8
  18. package/lib/Resource/resourceComponents.d.ts +4 -1
  19. package/lib/Resource/resourceComponents.js +14 -13
  20. package/lib/TreeStructure/FolderItems.d.ts +1 -1
  21. package/lib/TreeStructure/FolderItems.js +7 -7
  22. package/lib/TreeStructure/TreeStructure.d.ts +6 -1
  23. package/lib/TreeStructure/TreeStructure.js +16 -9
  24. package/lib/TreeStructure/TreeStructure.types.d.ts +3 -1
  25. package/lib/locale/messages-en.d.ts +1 -0
  26. package/lib/locale/messages-en.js +1 -0
  27. package/lib/locale/messages-nb.d.ts +1 -0
  28. package/lib/locale/messages-nb.js +1 -0
  29. package/lib/locale/messages-nn.d.ts +1 -0
  30. package/lib/locale/messages-nn.js +1 -0
  31. package/lib/locale/messages-se.d.ts +1 -0
  32. package/lib/locale/messages-se.js +1 -0
  33. package/lib/locale/messages-sma.d.ts +1 -0
  34. package/lib/locale/messages-sma.js +1 -0
  35. package/package.json +5 -5
  36. package/src/MyNdla/Resource/Folder.tsx +5 -5
  37. package/src/Resource/BlockResource.tsx +7 -6
  38. package/src/Resource/ListResource.tsx +7 -6
  39. package/src/Resource/resourceComponents.tsx +5 -1
  40. package/src/TreeStructure/FolderItems.tsx +3 -2
  41. package/src/TreeStructure/TreeStructure.tsx +21 -7
  42. package/src/TreeStructure/TreeStructure.types.ts +3 -1
  43. package/src/locale/messages-en.ts +1 -0
  44. package/src/locale/messages-nb.ts +1 -0
  45. package/src/locale/messages-nn.ts +1 -0
  46. package/src/locale/messages-se.ts +1 -0
  47. package/src/locale/messages-sma.ts +1 -0
@@ -70,11 +70,11 @@ exports.MAX_LEVEL_FOR_FOLDERS = MAX_LEVEL_FOR_FOLDERS;
70
70
  var StyledLabel = (0, _styledBase["default"])("label", {
71
71
  target: "e1dg1gdn0",
72
72
  label: "StyledLabel"
73
- })("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":"AAwBgC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport { uniq } from 'lodash';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0]?.id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      setOpenFolders((prev) => {\n        return uniq([...defaultOpenFolders, ...prev]);\n      });\n    }\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    if (openFolders.includes(id)) {\n      // Did we just closed a folder with a marked folder inside it?\n      // If so, we need to mark the folder we just closed.\n      if (markedFolderId) {\n        const closingFolderPath = getPathOfFolder(data, id);\n        const markedFolderPath = getPathOfFolder(data, markedFolderId);\n        const markedFolderIsSubPath = closingFolderPath.every(\n          (folderId, _index) => markedFolderPath[_index] === folderId,\n        );\n        if (markedFolderIsSubPath) {\n          setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n        }\n      }\n      setOpenFolders(openFolders.filter((folder) => folder !== id));\n    } else {\n      setOpenFolders(uniq([...openFolders, id]));\n    }\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      onNewFolder({ ...newFolder, value }).then((newFolderId) => {\n        if (newFolderId) {\n          setMarkedFolderId(newFolderId);\n          setFocusedFolderId(newFolderId);\n          // Open current folder in case it was closed..\n\n          if (newFolder.parentId) {\n            setOpenFolders(uniq([...openFolders, newFolder.parentId]));\n          }\n        }\n      });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <AddButton\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}\n            />\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
73
+ })("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":"AAwBgC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport { uniq } from 'lodash';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n  maximumLevelsOfFoldersAllowed,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0]?.id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      setOpenFolders((prev) => {\n        return uniq([...defaultOpenFolders, ...prev]);\n      });\n    }\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    if (openFolders.includes(id)) {\n      // Did we just closed a folder with a marked folder inside it?\n      // If so, we need to mark the folder we just closed.\n      if (markedFolderId) {\n        const closingFolderPath = getPathOfFolder(data, id);\n        const markedFolderPath = getPathOfFolder(data, markedFolderId);\n        const markedFolderIsSubPath = closingFolderPath.every(\n          (folderId, _index) => markedFolderPath[_index] === folderId,\n        );\n        if (markedFolderIsSubPath) {\n          setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n        }\n      }\n      setOpenFolders(openFolders.filter((folder) => folder !== id));\n    } else {\n      setOpenFolders(uniq([...openFolders, id]));\n    }\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      onNewFolder({ ...newFolder, value }).then((newFolderId) => {\n        if (newFolderId) {\n          setMarkedFolderId(newFolderId);\n          setFocusedFolderId(newFolderId);\n          // Open current folder in case it was closed..\n\n          if (newFolder.parentId) {\n            setOpenFolders(uniq([...openFolders, newFolder.parentId]));\n          }\n        }\n      });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  const paths = getPathOfFolder(data, markedFolderId || '');\n  const canAddFolder = editable && paths.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      {label && <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>}\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n          maximumLevelsOfFoldersAllowed={maximumLevelsOfFoldersAllowed}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={\n              canAddFolder\n                ? t('myNdla.newFolderUnder', {\n                    folderName: getFolderName(data, markedFolderId),\n                  })\n                : t('myNdla.maxFoldersAlreadyAdded')\n            }>\n            <AddButton\n              disabled={!canAddFolder}\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}>\n              {t('myNdla.newFolder')}\n            </AddButton>\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nTreeStructure.defaultProps = {\n  maximumLevelsOfFoldersAllowed: MAX_LEVEL_FOR_FOLDERS,\n};\n\nexport default TreeStructure;\n"]} */"));
74
74
  var AddFolderWrapper = (0, _styledBase["default"])("div", {
75
75
  target: "e1dg1gdn1",
76
76
  label: "AddFolderWrapper"
77
- })("display:flex;margin-top:", _core.spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA4BmC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport { uniq } from 'lodash';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0]?.id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      setOpenFolders((prev) => {\n        return uniq([...defaultOpenFolders, ...prev]);\n      });\n    }\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    if (openFolders.includes(id)) {\n      // Did we just closed a folder with a marked folder inside it?\n      // If so, we need to mark the folder we just closed.\n      if (markedFolderId) {\n        const closingFolderPath = getPathOfFolder(data, id);\n        const markedFolderPath = getPathOfFolder(data, markedFolderId);\n        const markedFolderIsSubPath = closingFolderPath.every(\n          (folderId, _index) => markedFolderPath[_index] === folderId,\n        );\n        if (markedFolderIsSubPath) {\n          setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n        }\n      }\n      setOpenFolders(openFolders.filter((folder) => folder !== id));\n    } else {\n      setOpenFolders(uniq([...openFolders, id]));\n    }\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      onNewFolder({ ...newFolder, value }).then((newFolderId) => {\n        if (newFolderId) {\n          setMarkedFolderId(newFolderId);\n          setFocusedFolderId(newFolderId);\n          // Open current folder in case it was closed..\n\n          if (newFolder.parentId) {\n            setOpenFolders(uniq([...openFolders, newFolder.parentId]));\n          }\n        }\n      });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <AddButton\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}\n            />\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
77
+ })("display:flex;margin-top:", _core.spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA4BmC","file":"TreeStructure.tsx","sourcesContent":["/**\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useState, useRef, useMemo } from 'react';\nimport { uuid } from '@ndla/util';\nimport { AddButton } from '@ndla/button';\nimport Tooltip from '@ndla/tooltip';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport { spacing, fonts } from '@ndla/core';\nimport { uniq } from 'lodash';\nimport TreeStructureStyledWrapper from './TreeStructureWrapper';\nimport FolderItems from './FolderItems';\nimport { getIdPathsOfFolder, getPathOfFolder, getFolderName } from './helperFunctions';\nimport keyboardNavigation, { KEYBOARD_KEYS_OF_INTEREST } from './keyboardNavigation/keyboardNavigation';\nimport { NewFolderProps, TreeStructureProps } from './TreeStructure.types';\n\nexport const MAX_LEVEL_FOR_FOLDERS = 4;\n\nconst StyledLabel = styled.label`\n  font-weight: ${fonts.weight.semibold};\n`;\n\nconst AddFolderWrapper = styled.div`\n  display: flex;\n  margin-top: ${spacing.xsmall};\n`;\n\nconst TreeStructure = ({\n  data,\n  label,\n  editable,\n  loading,\n  onNewFolder,\n  openOnFolderClick,\n  framed,\n  folderIdMarkedByDefault,\n  defaultOpenFolders,\n  folderChild,\n  maximumLevelsOfFoldersAllowed,\n}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<string[]>(defaultOpenFolders || []);\n  const [focusedFolderId, setFocusedFolderId] = useState<string | undefined>();\n  const [markedFolderId, setMarkedFolderId] = useState<string | undefined>(folderIdMarkedByDefault || data[0]?.id);\n  const treestructureRef = useRef<HTMLDivElement>(null);\n  const wrapperRef = useRef<HTMLDivElement>(null);\n  const rootLevelId = useMemo(() => uuid(), []); // TODO: use useId hook when we update to React 18\n\n  useEffect(() => {\n    if (defaultOpenFolders) {\n      setOpenFolders((prev) => {\n        return uniq([...defaultOpenFolders, ...prev]);\n      });\n    }\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    if (openFolders.includes(id)) {\n      // Did we just closed a folder with a marked folder inside it?\n      // If so, we need to mark the folder we just closed.\n      if (markedFolderId) {\n        const closingFolderPath = getPathOfFolder(data, id);\n        const markedFolderPath = getPathOfFolder(data, markedFolderId);\n        const markedFolderIsSubPath = closingFolderPath.every(\n          (folderId, _index) => markedFolderPath[_index] === folderId,\n        );\n        if (markedFolderIsSubPath) {\n          setMarkedFolderId(closingFolderPath[closingFolderPath.length - 1]);\n        }\n      }\n      setOpenFolders(openFolders.filter((folder) => folder !== id));\n    } else {\n      setOpenFolders(uniq([...openFolders, id]));\n    }\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = (value: string) => {\n    if (newFolder) {\n      // We would like to create a new folder with the name of value.\n      // Its location in structure is based on newFolder object\n      onNewFolder({ ...newFolder, value }).then((newFolderId) => {\n        if (newFolderId) {\n          setMarkedFolderId(newFolderId);\n          setFocusedFolderId(newFolderId);\n          // Open current folder in case it was closed..\n\n          if (newFolder.parentId) {\n            setOpenFolders(uniq([...openFolders, newFolder.parentId]));\n          }\n        }\n      });\n    }\n  };\n\n  const onCancelNewFolder = () => {\n    setNewFolder(undefined);\n  };\n\n  const onMarkFolder = (id: string) => {\n    setMarkedFolderId(id);\n    setFocusedFolderId(id);\n  };\n\n  const paths = getPathOfFolder(data, markedFolderId || '');\n  const canAddFolder = editable && paths.length < (maximumLevelsOfFoldersAllowed || 1);\n\n  return (\n    <div\n      ref={treestructureRef}\n      onKeyDown={(e) => {\n        if (wrapperRef.current?.contains(document.activeElement) && KEYBOARD_KEYS_OF_INTEREST.includes(e.key)) {\n          keyboardNavigation({\n            e,\n            data,\n            setFocusedFolderId,\n            focusedFolderId,\n            onToggleOpen,\n            openFolders,\n          });\n        }\n      }}>\n      {label && <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>}\n      <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label=\"Menu tree\" role=\"tree\" framed={framed}>\n        <FolderItems\n          idPaths={[]}\n          data={data}\n          editable={editable}\n          onToggleOpen={onToggleOpen}\n          newFolder={newFolder}\n          onCreateNewFolder={onCreateNewFolder}\n          onCancelNewFolder={onCancelNewFolder}\n          onSaveNewFolder={onSaveNewFolder}\n          openFolders={openFolders}\n          markedFolderId={markedFolderId}\n          onMarkFolder={onMarkFolder}\n          openOnFolderClick={openOnFolderClick}\n          loading={loading}\n          focusedFolderId={focusedFolderId}\n          setFocusedFolderId={setFocusedFolderId}\n          firstLevel\n          folderChild={folderChild}\n          maximumLevelsOfFoldersAllowed={maximumLevelsOfFoldersAllowed}\n        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={\n              canAddFolder\n                ? t('myNdla.newFolderUnder', {\n                    folderName: getFolderName(data, markedFolderId),\n                  })\n                : t('myNdla.maxFoldersAlreadyAdded')\n            }>\n            <AddButton\n              disabled={!canAddFolder}\n              aria-label={t('myNdla.newFolder')}\n              onClick={() => {\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}>\n              {t('myNdla.newFolder')}\n            </AddButton>\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nTreeStructure.defaultProps = {\n  maximumLevelsOfFoldersAllowed: MAX_LEVEL_FOR_FOLDERS,\n};\n\nexport default TreeStructure;\n"]} */"));
78
78
 
79
79
  var TreeStructure = function TreeStructure(_ref) {
80
80
  var _data$;
@@ -88,7 +88,8 @@ var TreeStructure = function TreeStructure(_ref) {
88
88
  framed = _ref.framed,
89
89
  folderIdMarkedByDefault = _ref.folderIdMarkedByDefault,
90
90
  defaultOpenFolders = _ref.defaultOpenFolders,
91
- folderChild = _ref.folderChild;
91
+ folderChild = _ref.folderChild,
92
+ maximumLevelsOfFoldersAllowed = _ref.maximumLevelsOfFoldersAllowed;
92
93
 
93
94
  var _useTranslation = (0, _reactI18next.useTranslation)(),
94
95
  t = _useTranslation.t;
@@ -188,6 +189,8 @@ var TreeStructure = function TreeStructure(_ref) {
188
189
  setFocusedFolderId(id);
189
190
  };
190
191
 
192
+ var paths = (0, _helperFunctions.getPathOfFolder)(data, markedFolderId || '');
193
+ var canAddFolder = editable && paths.length < (maximumLevelsOfFoldersAllowed || 1);
191
194
  return (0, _core2.jsx)("div", {
192
195
  ref: treestructureRef,
193
196
  onKeyDown: function onKeyDown(e) {
@@ -204,7 +207,7 @@ var TreeStructure = function TreeStructure(_ref) {
204
207
  });
205
208
  }
206
209
  }
207
- }, (0, _core2.jsx)(StyledLabel, {
210
+ }, label && (0, _core2.jsx)(StyledLabel, {
208
211
  htmlFor: rootLevelId
209
212
  }, label), (0, _core2.jsx)(_TreeStructureWrapper["default"], {
210
213
  ref: wrapperRef,
@@ -229,23 +232,27 @@ var TreeStructure = function TreeStructure(_ref) {
229
232
  focusedFolderId: focusedFolderId,
230
233
  setFocusedFolderId: setFocusedFolderId,
231
234
  firstLevel: true,
232
- folderChild: folderChild
235
+ folderChild: folderChild,
236
+ maximumLevelsOfFoldersAllowed: maximumLevelsOfFoldersAllowed
233
237
  })), editable && (0, _core2.jsx)(AddFolderWrapper, null, (0, _core2.jsx)(_tooltip["default"], {
234
- tooltip: t('myNdla.newFolderUnder', {
238
+ tooltip: canAddFolder ? t('myNdla.newFolderUnder', {
235
239
  folderName: (0, _helperFunctions.getFolderName)(data, markedFolderId)
236
- })
240
+ }) : t('myNdla.maxFoldersAlreadyAdded')
237
241
  }, (0, _core2.jsx)(_button.AddButton, {
242
+ disabled: !canAddFolder,
238
243
  "aria-label": t('myNdla.newFolder'),
239
244
  onClick: function onClick() {
240
- var paths = (0, _helperFunctions.getPathOfFolder)(data, markedFolderId || '');
241
245
  var idPaths = (0, _helperFunctions.getIdPathsOfFolder)(data, markedFolderId || '');
242
246
  setNewFolder({
243
247
  idPaths: idPaths,
244
248
  parentId: paths[paths.length - 1]
245
249
  });
246
250
  }
247
- }))));
251
+ }, t('myNdla.newFolder')))));
248
252
  };
249
253
 
254
+ TreeStructure.defaultProps = {
255
+ maximumLevelsOfFoldersAllowed: MAX_LEVEL_FOR_FOLDERS
256
+ };
250
257
  var _default = TreeStructure;
251
258
  exports["default"] = _default;
@@ -29,7 +29,7 @@ interface CommonFolderProps {
29
29
  }
30
30
  export interface TreeStructureProps extends CommonFolderProps {
31
31
  framed?: boolean;
32
- label: string;
32
+ label?: string;
33
33
  folderIdMarkedByDefault?: string;
34
34
  onNewFolder: (props: {
35
35
  value: string;
@@ -38,6 +38,7 @@ export interface TreeStructureProps extends CommonFolderProps {
38
38
  }) => Promise<string>;
39
39
  defaultOpenFolders?: string[];
40
40
  folderChild?: FolderChildFuncType;
41
+ maximumLevelsOfFoldersAllowed: number;
41
42
  }
42
43
  export declare type onCreateNewFolderProp = ({ idPaths, parentId, }: {
43
44
  idPaths: number[];
@@ -62,5 +63,6 @@ export interface FolderItemsProps extends CommonFolderProps {
62
63
  keyNavigationFocusIsCreateFolderButton?: boolean;
63
64
  icon?: ReactNode;
64
65
  folderChild?: FolderChildFuncType;
66
+ maximumLevelsOfFoldersAllowed: number;
65
67
  }
66
68
  export {};
@@ -132,6 +132,7 @@ declare const messages: {
132
132
  delete: string;
133
133
  };
134
134
  createFolder: string;
135
+ maxFoldersAlreadyAdded: string;
135
136
  newFolder: {
136
137
  placeholder: string;
137
138
  defaultName: string;
@@ -34,6 +34,7 @@ var messages = _objectSpread(_objectSpread({
34
34
  "delete": 'Delete'
35
35
  },
36
36
  createFolder: 'Create folder',
37
+ maxFoldersAlreadyAdded: 'Maximum subfolders reached',
37
38
  newFolder: {
38
39
  placeholder: 'Add foldername',
39
40
  defaultName: 'New folder'
@@ -132,6 +132,7 @@ declare const messages: {
132
132
  delete: string;
133
133
  };
134
134
  createFolder: string;
135
+ maxFoldersAlreadyAdded: string;
135
136
  newFolder: {
136
137
  placeholder: string;
137
138
  defaultName: string;
@@ -34,6 +34,7 @@ var messages = _objectSpread(_objectSpread({
34
34
  "delete": 'Slett'
35
35
  },
36
36
  createFolder: 'Lag mappe',
37
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
37
38
  newFolder: {
38
39
  placeholder: 'Skriv navn på mappe',
39
40
  defaultName: 'Ny mappe'
@@ -132,6 +132,7 @@ declare const messages: {
132
132
  delete: string;
133
133
  };
134
134
  createFolder: string;
135
+ maxFoldersAlreadyAdded: string;
135
136
  newFolder: {
136
137
  placeholder: string;
137
138
  defaultName: string;
@@ -34,6 +34,7 @@ var messages = _objectSpread(_objectSpread({
34
34
  "delete": 'Slett'
35
35
  },
36
36
  createFolder: 'Lag mappe',
37
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
37
38
  newFolder: {
38
39
  placeholder: 'Skriv namn på mappe',
39
40
  defaultName: 'Ny mappe'
@@ -132,6 +132,7 @@ declare const messages: {
132
132
  delete: string;
133
133
  };
134
134
  createFolder: string;
135
+ maxFoldersAlreadyAdded: string;
135
136
  newFolder: {
136
137
  placeholder: string;
137
138
  defaultName: string;
@@ -34,6 +34,7 @@ var messages = _objectSpread(_objectSpread({
34
34
  "delete": 'Slett'
35
35
  },
36
36
  createFolder: 'Lag mappe',
37
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
37
38
  newFolder: {
38
39
  placeholder: 'Skriv navn på mappe',
39
40
  defaultName: 'Ny mappe'
@@ -132,6 +132,7 @@ declare const messages: {
132
132
  delete: string;
133
133
  };
134
134
  createFolder: string;
135
+ maxFoldersAlreadyAdded: string;
135
136
  newFolder: {
136
137
  placeholder: string;
137
138
  defaultName: string;
@@ -34,6 +34,7 @@ var messages = _objectSpread(_objectSpread({
34
34
  "delete": 'Slett'
35
35
  },
36
36
  createFolder: 'Lag mappe',
37
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
37
38
  newFolder: {
38
39
  placeholder: 'Skriv navn på mappe',
39
40
  defaultName: 'Ny mappe'
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ndla/ui",
3
- "version": "18.0.1",
3
+ "version": "19.0.0",
4
4
  "description": "UI component library for NDLA.",
5
5
  "license": "GPL-3.0",
6
6
  "main": "lib/index.js",
@@ -31,15 +31,15 @@
31
31
  "types"
32
32
  ],
33
33
  "dependencies": {
34
- "@ndla/button": "^2.6.0",
34
+ "@ndla/button": "^3.0.0",
35
35
  "@ndla/carousel": "^1.2.11",
36
36
  "@ndla/core": "^2.3.0",
37
37
  "@ndla/hooks": "^1.1.4",
38
38
  "@ndla/icons": "^1.10.0",
39
39
  "@ndla/licenses": "^5.0.2",
40
40
  "@ndla/modal": "^1.2.12",
41
- "@ndla/notion": "^3.1.17",
42
- "@ndla/safelink": "^2.0.8",
41
+ "@ndla/notion": "^3.1.18",
42
+ "@ndla/safelink": "^2.0.9",
43
43
  "@ndla/switch": "^0.1.7",
44
44
  "@ndla/tabs": "^1.1.10",
45
45
  "@ndla/tooltip": "^2.1.2",
@@ -81,5 +81,5 @@
81
81
  "publishConfig": {
82
82
  "access": "public"
83
83
  },
84
- "gitHead": "841ff68015ec6032bed44a7c25eea2dfd9088a64"
84
+ "gitHead": "ae09299c27fc3f48f1b8f77ebe37c7aac511b2ad"
85
85
  }
@@ -7,14 +7,14 @@
7
7
  */
8
8
 
9
9
  import styled from '@emotion/styled';
10
- import React, { ReactNode } from 'react';
10
+ import React from 'react';
11
11
  import { FolderOutlined } from '@ndla/icons/contentType';
12
12
  import { FileDocumentOutline } from '@ndla/icons/common';
13
13
  import { fonts, spacing, colors, mq, breakpoints } from '@ndla/core';
14
14
  import { css } from '@emotion/core';
15
15
  import { useTranslation } from 'react-i18next';
16
16
  import SafeLink from '@ndla/safelink';
17
- import { MenuButton } from '@ndla/button';
17
+ import { MenuButton, MenuItemProps } from '@ndla/button';
18
18
 
19
19
  interface FolderIconWrapperProps {
20
20
  type?: LayoutType;
@@ -84,7 +84,7 @@ interface Props {
84
84
  description?: string;
85
85
  link: string;
86
86
  type: LayoutType;
87
- actionMenu?: ReactNode;
87
+ menuItems?: MenuItemProps[];
88
88
  }
89
89
 
90
90
  interface IconCountProps {
@@ -131,7 +131,7 @@ const IconCount = ({ type, count, layoutType }: IconCountProps) => {
131
131
 
132
132
  type LayoutType = 'list' | 'block';
133
133
 
134
- const Folder = ({ link, title, subFolders, subResources, type = 'list', actionMenu }: Props) => {
134
+ const Folder = ({ link, title, subFolders, subResources, type = 'list', menuItems }: Props) => {
135
135
  const { t } = useTranslation();
136
136
  return (
137
137
  <FolderWrapper to={link}>
@@ -141,7 +141,7 @@ const Folder = ({ link, title, subFolders, subResources, type = 'list', actionMe
141
141
  <FolderTitle>{title}</FolderTitle>
142
142
  <IconCount layoutType={type} type={'folder'} count={subFolders} />
143
143
  <IconCount layoutType={type} type={'resource'} count={subResources} />
144
- <MenuButton size="small" />
144
+ {menuItems && menuItems.length > 0 && <MenuButton size="small" menuItems={menuItems} />}
145
145
  </FolderWrapper>
146
146
  );
147
147
  };
@@ -7,11 +7,12 @@
7
7
  */
8
8
 
9
9
  import styled from '@emotion/styled';
10
- import React, { ReactNode } from 'react';
10
+ import React from 'react';
11
11
  import SafeLink from '@ndla/safelink';
12
12
  import { colors, fonts, spacing } from '@ndla/core';
13
+ import { MenuButton, MenuItemProps } from '@ndla/button';
13
14
  import Image from '../Image';
14
- import { CompressedTags, ResourceImageProps, ResourceTitle, Row, TopicList } from './resourceComponents';
15
+ import { CompressedTagList, ResourceImageProps, ResourceTitle, Row, TopicList } from './resourceComponents';
15
16
 
16
17
  interface BlockResourceProps {
17
18
  link: string;
@@ -20,7 +21,7 @@ interface BlockResourceProps {
20
21
  topics: string[];
21
22
  tags?: string[];
22
23
  description?: string;
23
- actionMenu?: ReactNode;
24
+ menuItems?: MenuItemProps[];
24
25
  }
25
26
 
26
27
  const BlockElementWrapper = styled(SafeLink)`
@@ -77,7 +78,7 @@ const ImageWrapper = styled.div`
77
78
  }
78
79
  `;
79
80
 
80
- const BlockResource = ({ link, title, tags, resourceImage, topics, description, actionMenu }: BlockResourceProps) => {
81
+ const BlockResource = ({ link, title, tags, resourceImage, topics, description, menuItems }: BlockResourceProps) => {
81
82
  return (
82
83
  <BlockElementWrapper to={link}>
83
84
  <ImageWrapper>
@@ -90,8 +91,8 @@ const BlockResource = ({ link, title, tags, resourceImage, topics, description,
90
91
  <TopicList topics={topics} />
91
92
  <BlockDescription>{description}</BlockDescription>
92
93
  <RightRow>
93
- {tags && CompressedTags(tags)}
94
- {actionMenu}
94
+ {tags && <CompressedTagList tags={tags} />}
95
+ {menuItems && menuItems.length > 0 && <MenuButton size="small" menuItems={menuItems} />}
95
96
  </RightRow>
96
97
  </BlockInfoWrapper>
97
98
  </BlockElementWrapper>
@@ -7,11 +7,12 @@
7
7
  */
8
8
 
9
9
  import styled from '@emotion/styled';
10
- import React, { ReactNode } from 'react';
10
+ import React from 'react';
11
11
  import SafeLink from '@ndla/safelink';
12
12
  import { fonts, spacing, colors, breakpoints, mq } from '@ndla/core';
13
+ import { MenuButton, MenuItemProps } from '@ndla/button';
13
14
  import Image from '../Image';
14
- import { CompressedTags, ResourceImageProps, ResourceTitle, TopicList } from './resourceComponents';
15
+ import { CompressedTagList, ResourceImageProps, ResourceTitle, TopicList } from './resourceComponents';
15
16
 
16
17
  const ResourceDescription = styled.p`
17
18
  grid-area: description;
@@ -109,10 +110,10 @@ export interface ListResourceProps {
109
110
  topics: string[];
110
111
  tags?: string[];
111
112
  description?: string;
112
- actionMenu?: ReactNode;
113
+ menuItems?: MenuItemProps[];
113
114
  }
114
115
 
115
- const ListResource = ({ link, title, tags, resourceImage, topics, description, actionMenu }: ListResourceProps) => {
116
+ const ListResource = ({ link, title, tags, resourceImage, topics, description, menuItems }: ListResourceProps) => {
116
117
  const showDescription = description !== undefined;
117
118
 
118
119
  return (
@@ -126,8 +127,8 @@ const ListResource = ({ link, title, tags, resourceImage, topics, description, a
126
127
  </TopicAndTitle>
127
128
  {showDescription && <ResourceDescription>{description}</ResourceDescription>}
128
129
  <TagsandActionMenu>
129
- {tags && CompressedTags(tags)}
130
- {actionMenu}
130
+ {tags && <CompressedTagList tags={tags} />}
131
+ {menuItems && menuItems.length > 0 && <MenuButton size="small" menuItems={menuItems} />}
131
132
  </TagsandActionMenu>
132
133
  </ResourceWrapper>
133
134
  );
@@ -103,7 +103,11 @@ export const TagList = ({ tags }: TagListProps) => {
103
103
  );
104
104
  };
105
105
 
106
- export const CompressedTags = (tags: string[]) => {
106
+ interface CompressedTagListProps {
107
+ tags: string[];
108
+ }
109
+
110
+ export const CompressedTagList = ({ tags }: CompressedTagListProps) => {
107
111
  const visibleTags = tags.slice(0, 3);
108
112
  const remainingTags = tags.slice(3, tags.length).map((tag) => {
109
113
  return {
@@ -12,7 +12,6 @@ import { animations, spacing } from '@ndla/core';
12
12
  import FolderItem from './FolderItem';
13
13
  import FolderNameInput from './FolderNameInput';
14
14
  import { FolderItemsProps } from './TreeStructure.types';
15
- import { MAX_LEVEL_FOR_FOLDERS } from './TreeStructure';
16
15
 
17
16
  const StyledUL = styled.ul<{ firstLevel?: boolean }>`
18
17
  ${animations.fadeInLeft(animations.durations.fast)};
@@ -49,6 +48,7 @@ const FolderItems = ({
49
48
  setFocusedFolderId,
50
49
  firstLevel,
51
50
  folderChild,
51
+ maximumLevelsOfFoldersAllowed,
52
52
  }: FolderItemsProps) => (
53
53
  <StyledUL role="group" firstLevel={firstLevel}>
54
54
  {data.map(({ name, data: dataChildren, id, url, icon }, _index) => {
@@ -69,7 +69,7 @@ const FolderItems = ({
69
69
  focusedFolderId={focusedFolderId}
70
70
  onToggleOpen={onToggleOpen}
71
71
  onMarkFolder={onMarkFolder}
72
- hideArrow={dataChildren?.length === 0 || newIdPaths.length >= MAX_LEVEL_FOR_FOLDERS}
72
+ hideArrow={dataChildren?.length === 0 || newIdPaths.length >= maximumLevelsOfFoldersAllowed}
73
73
  noPaddingWhenArrowIsHidden={editable && firstLevel && dataChildren?.length === 0}
74
74
  setFocusedFolderId={setFocusedFolderId}
75
75
  folderChild={folderChild}
@@ -101,6 +101,7 @@ const FolderItems = ({
101
101
  setFocusedFolderId={setFocusedFolderId}
102
102
  firstLevel={false}
103
103
  folderChild={folderChild}
104
+ maximumLevelsOfFoldersAllowed={maximumLevelsOfFoldersAllowed}
104
105
  />
105
106
  )}
106
107
  </StyledLI>
@@ -42,6 +42,7 @@ const TreeStructure = ({
42
42
  folderIdMarkedByDefault,
43
43
  defaultOpenFolders,
44
44
  folderChild,
45
+ maximumLevelsOfFoldersAllowed,
45
46
  }: TreeStructureProps) => {
46
47
  const { t } = useTranslation();
47
48
  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();
@@ -117,6 +118,9 @@ const TreeStructure = ({
117
118
  setFocusedFolderId(id);
118
119
  };
119
120
 
121
+ const paths = getPathOfFolder(data, markedFolderId || '');
122
+ const canAddFolder = editable && paths.length < (maximumLevelsOfFoldersAllowed || 1);
123
+
120
124
  return (
121
125
  <div
122
126
  ref={treestructureRef}
@@ -132,7 +136,7 @@ const TreeStructure = ({
132
136
  });
133
137
  }
134
138
  }}>
135
- <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>
139
+ {label && <StyledLabel htmlFor={rootLevelId}>{label}</StyledLabel>}
136
140
  <TreeStructureStyledWrapper ref={wrapperRef} id={rootLevelId} aria-label="Menu tree" role="tree" framed={framed}>
137
141
  <FolderItems
138
142
  idPaths={[]}
@@ -152,22 +156,28 @@ const TreeStructure = ({
152
156
  setFocusedFolderId={setFocusedFolderId}
153
157
  firstLevel
154
158
  folderChild={folderChild}
159
+ maximumLevelsOfFoldersAllowed={maximumLevelsOfFoldersAllowed}
155
160
  />
156
161
  </TreeStructureStyledWrapper>
157
162
  {editable && (
158
163
  <AddFolderWrapper>
159
164
  <Tooltip
160
- tooltip={t('myNdla.newFolderUnder', {
161
- folderName: getFolderName(data, markedFolderId),
162
- })}>
165
+ tooltip={
166
+ canAddFolder
167
+ ? t('myNdla.newFolderUnder', {
168
+ folderName: getFolderName(data, markedFolderId),
169
+ })
170
+ : t('myNdla.maxFoldersAlreadyAdded')
171
+ }>
163
172
  <AddButton
173
+ disabled={!canAddFolder}
164
174
  aria-label={t('myNdla.newFolder')}
165
175
  onClick={() => {
166
- const paths = getPathOfFolder(data, markedFolderId || '');
167
176
  const idPaths = getIdPathsOfFolder(data, markedFolderId || '');
168
177
  setNewFolder({ idPaths, parentId: paths[paths.length - 1] });
169
- }}
170
- />
178
+ }}>
179
+ {t('myNdla.newFolder')}
180
+ </AddButton>
171
181
  </Tooltip>
172
182
  </AddFolderWrapper>
173
183
  )}
@@ -175,4 +185,8 @@ const TreeStructure = ({
175
185
  );
176
186
  };
177
187
 
188
+ TreeStructure.defaultProps = {
189
+ maximumLevelsOfFoldersAllowed: MAX_LEVEL_FOR_FOLDERS,
190
+ };
191
+
178
192
  export default TreeStructure;
@@ -34,11 +34,12 @@ interface CommonFolderProps {
34
34
 
35
35
  export interface TreeStructureProps extends CommonFolderProps {
36
36
  framed?: boolean;
37
- label: string;
37
+ label?: string;
38
38
  folderIdMarkedByDefault?: string;
39
39
  onNewFolder: (props: { value: string; parentId?: string; idPaths: number[] }) => Promise<string>;
40
40
  defaultOpenFolders?: string[];
41
41
  folderChild?: FolderChildFuncType;
42
+ maximumLevelsOfFoldersAllowed: number;
42
43
  }
43
44
 
44
45
  export type onCreateNewFolderProp = ({
@@ -70,4 +71,5 @@ export interface FolderItemsProps extends CommonFolderProps {
70
71
  keyNavigationFocusIsCreateFolderButton?: boolean;
71
72
  icon?: ReactNode;
72
73
  folderChild?: FolderChildFuncType;
74
+ maximumLevelsOfFoldersAllowed: number;
73
75
  }
@@ -20,6 +20,7 @@ const messages = {
20
20
  delete: 'Delete',
21
21
  },
22
22
  createFolder: 'Create folder',
23
+ maxFoldersAlreadyAdded: 'Maximum subfolders reached',
23
24
  newFolder: {
24
25
  placeholder: 'Add foldername',
25
26
  defaultName: 'New folder',
@@ -20,6 +20,7 @@ const messages = {
20
20
  delete: 'Slett',
21
21
  },
22
22
  createFolder: 'Lag mappe',
23
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
23
24
  newFolder: {
24
25
  placeholder: 'Skriv navn på mappe',
25
26
  defaultName: 'Ny mappe',
@@ -20,6 +20,7 @@ const messages = {
20
20
  delete: 'Slett',
21
21
  },
22
22
  createFolder: 'Lag mappe',
23
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
23
24
  newFolder: {
24
25
  placeholder: 'Skriv namn på mappe',
25
26
  defaultName: 'Ny mappe',
@@ -20,6 +20,7 @@ const messages = {
20
20
  delete: 'Slett',
21
21
  },
22
22
  createFolder: 'Lag mappe',
23
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
23
24
  newFolder: {
24
25
  placeholder: 'Skriv navn på mappe',
25
26
  defaultName: 'Ny mappe',
@@ -20,6 +20,7 @@ const messages = {
20
20
  delete: 'Slett',
21
21
  },
22
22
  createFolder: 'Lag mappe',
23
+ maxFoldersAlreadyAdded: 'Maks nivå av undermapper nådd',
23
24
  newFolder: {
24
25
  placeholder: 'Skriv navn på mappe',
25
26
  defaultName: 'Ny mappe',