@ndla/ui 15.2.0 → 16.1.1

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 (58) hide show
  1. package/es/Article/ArticleContent.js +2 -0
  2. package/es/FileList/File.js +7 -12
  3. package/es/Filter/FilterButtons.js +13 -13
  4. package/es/NDLAFilm/CategorySelect.js +51 -25
  5. package/es/NDLAFilm/FilmMovieList.js +14 -10
  6. package/es/NDLAFilm/FilmMovieSearch.js +23 -10
  7. package/es/NDLAFilm/FilmSlideshow.js +117 -19
  8. package/es/NDLAFilm/MovieGrid.js +23 -14
  9. package/es/NDLAFilm/NavigationArrow.js +33 -7
  10. package/es/NDLAFilm/SlideshowIndicator.js +27 -11
  11. package/es/NDLAFilm/filmStyles.js +23 -0
  12. package/es/RadioButtonGroup/RadioButtonGroup.js +28 -9
  13. package/es/Resource/resourceComponents.js +17 -9
  14. package/es/TreeStructure/TreeStructure.js +6 -9
  15. package/es/all.css +1 -1
  16. package/lib/Article/ArticleContent.js +3 -0
  17. package/lib/FileList/File.js +10 -14
  18. package/lib/Filter/FilterButtons.js +13 -13
  19. package/lib/Frontpage/FrontpageMultidisciplinarySubject.d.ts +5 -5
  20. package/lib/Frontpage/illustrations/FrontpageIllustrations.d.ts +2 -2
  21. package/lib/MultidisciplinarySubject/Illustrations.d.ts +7 -7
  22. package/lib/NDLAFilm/CategorySelect.js +52 -30
  23. package/lib/NDLAFilm/FilmMovieList.js +17 -13
  24. package/lib/NDLAFilm/FilmMovieSearch.js +29 -17
  25. package/lib/NDLAFilm/FilmSlideshow.js +113 -24
  26. package/lib/NDLAFilm/MovieGrid.js +26 -16
  27. package/lib/NDLAFilm/NavigationArrow.d.ts +5 -1
  28. package/lib/NDLAFilm/NavigationArrow.js +34 -10
  29. package/lib/NDLAFilm/SlideshowIndicator.js +34 -13
  30. package/lib/NDLAFilm/filmStyles.d.ts +8 -0
  31. package/lib/NDLAFilm/filmStyles.js +38 -0
  32. package/lib/RadioButtonGroup/RadioButtonGroup.js +28 -13
  33. package/lib/Resource/resourceComponents.d.ts +2 -2
  34. package/lib/Resource/resourceComponents.js +17 -9
  35. package/lib/Search/ContentTypeResultStyles.d.ts +3 -3
  36. package/lib/SearchTypeResult/ActiveFilterContent.d.ts +1 -1
  37. package/lib/TreeStructure/TreeStructure.js +6 -9
  38. package/lib/all.css +1 -1
  39. package/package.json +7 -7
  40. package/src/Article/ArticleContent.tsx +2 -0
  41. package/src/FileList/File.tsx +8 -11
  42. package/src/Filter/FilterButtons.tsx +3 -4
  43. package/src/NDLAFilm/CategorySelect.tsx +110 -23
  44. package/src/NDLAFilm/FilmMovieList.tsx +13 -11
  45. package/src/NDLAFilm/FilmMovieSearch.tsx +45 -14
  46. package/src/NDLAFilm/FilmSlideshow.tsx +186 -25
  47. package/src/NDLAFilm/MovieGrid.tsx +33 -25
  48. package/src/NDLAFilm/NavigationArrow.tsx +76 -10
  49. package/src/NDLAFilm/SlideshowIndicator.tsx +53 -11
  50. package/src/NDLAFilm/component.film-movielist.scss +0 -46
  51. package/src/NDLAFilm/filmStyles.ts +33 -0
  52. package/src/RadioButtonGroup/RadioButtonGroup.tsx +82 -11
  53. package/src/Resource/resourceComponents.tsx +1 -3
  54. package/src/TreeStructure/TreeStructure.tsx +5 -11
  55. package/src/main.scss +0 -3
  56. package/src/NDLAFilm/component.film-moviesearch.scss +0 -127
  57. package/src/NDLAFilm/component.film-slideshow.scss +0 -258
  58. package/src/RadioButtonGroup/component.radio-button-group.scss +0 -67
@@ -31,7 +31,7 @@ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
31
31
  */
32
32
  import React, { useEffect, useState, useRef, useMemo } from 'react';
33
33
  import { uuid } from '@ndla/util';
34
- import Button from '@ndla/button';
34
+ import { AddButton } from '@ndla/button';
35
35
  import Tooltip from '@ndla/tooltip';
36
36
  import { useTranslation } from 'react-i18next';
37
37
  import { spacing, fonts } from '@ndla/core';
@@ -45,12 +45,12 @@ export var MAX_LEVEL_FOR_FOLDERS = 4;
45
45
  var StyledLabel = _styled("label", {
46
46
  target: "e1dg1gdn0",
47
47
  label: "StyledLabel"
48
- })("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":"AAuBgC","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 Button 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 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}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<Set<string>>(new Set(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    setOpenFolders((prev) => {\n      defaultOpenFolders?.forEach((id) => prev.add(id));\n      return new Set(prev);\n    });\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    setOpenFolders((prev) => {\n      if (prev.has(id)) {\n        prev.delete(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      } else {\n        prev.add(id);\n      }\n      return new Set(prev);\n    });\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = async (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      const newFolderId = await onNewFolder({ ...newFolder, value });\n      if (newFolderId) {\n        setMarkedFolderId(newFolderId);\n        setFocusedFolderId(newFolderId);\n        // Open current folder in case it was closed..\n        setOpenFolders((prev) => {\n          if (newFolder.parentId) {\n            prev.add(newFolder.parentId);\n          }\n          return new Set(prev);\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 disableAddFolderButton =\n    markedFolderId === undefined || getPathOfFolder(data, markedFolderId).length >= MAX_LEVEL_FOR_FOLDERS;\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        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <Button\n              size=\"small\"\n              light\n              disabled={disableAddFolderButton}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}>\n              {t('myNdla.newFolder')}\n            </Button>\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
48
+ })("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":"AAuBgC","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 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}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<Set<string>>(new Set(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    setOpenFolders((prev) => {\n      defaultOpenFolders?.forEach((id) => prev.add(id));\n      return new Set(prev);\n    });\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    setOpenFolders((prev) => {\n      if (prev.has(id)) {\n        prev.delete(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      } else {\n        prev.add(id);\n      }\n      return new Set(prev);\n    });\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = async (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      const newFolderId = await onNewFolder({ ...newFolder, value });\n      if (newFolderId) {\n        setMarkedFolderId(newFolderId);\n        setFocusedFolderId(newFolderId);\n        // Open current folder in case it was closed..\n        setOpenFolders((prev) => {\n          if (newFolder.parentId) {\n            prev.add(newFolder.parentId);\n          }\n          return new Set(prev);\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        />\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"]} */"));
49
49
 
50
50
  var AddFolderWrapper = _styled("div", {
51
51
  target: "e1dg1gdn1",
52
52
  label: "AddFolderWrapper"
53
- })("display:flex;margin-top:", spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA2BmC","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 Button 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 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}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<Set<string>>(new Set(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    setOpenFolders((prev) => {\n      defaultOpenFolders?.forEach((id) => prev.add(id));\n      return new Set(prev);\n    });\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    setOpenFolders((prev) => {\n      if (prev.has(id)) {\n        prev.delete(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      } else {\n        prev.add(id);\n      }\n      return new Set(prev);\n    });\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = async (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      const newFolderId = await onNewFolder({ ...newFolder, value });\n      if (newFolderId) {\n        setMarkedFolderId(newFolderId);\n        setFocusedFolderId(newFolderId);\n        // Open current folder in case it was closed..\n        setOpenFolders((prev) => {\n          if (newFolder.parentId) {\n            prev.add(newFolder.parentId);\n          }\n          return new Set(prev);\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 disableAddFolderButton =\n    markedFolderId === undefined || getPathOfFolder(data, markedFolderId).length >= MAX_LEVEL_FOR_FOLDERS;\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        />\n      </TreeStructureStyledWrapper>\n      {editable && (\n        <AddFolderWrapper>\n          <Tooltip\n            tooltip={t('myNdla.newFolderUnder', {\n              folderName: getFolderName(data, markedFolderId),\n            })}>\n            <Button\n              size=\"small\"\n              light\n              disabled={disableAddFolderButton}\n              onClick={() => {\n                const paths = getPathOfFolder(data, markedFolderId || '');\n                const idPaths = getIdPathsOfFolder(data, markedFolderId || '');\n                setNewFolder({ idPaths, parentId: paths[paths.length - 1] });\n              }}>\n              {t('myNdla.newFolder')}\n            </Button>\n          </Tooltip>\n        </AddFolderWrapper>\n      )}\n    </div>\n  );\n};\n\nexport default TreeStructure;\n"]} */"));
53
+ })("display:flex;margin-top:", spacing.xsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeStructure.tsx"],"names":[],"mappings":"AA2BmC","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 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}: TreeStructureProps) => {\n  const { t } = useTranslation();\n  const [newFolder, setNewFolder] = useState<NewFolderProps | undefined>();\n  const [openFolders, setOpenFolders] = useState<Set<string>>(new Set(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    setOpenFolders((prev) => {\n      defaultOpenFolders?.forEach((id) => prev.add(id));\n      return new Set(prev);\n    });\n  }, [defaultOpenFolders]);\n\n  useEffect(() => {\n    if (!loading) {\n      setNewFolder(undefined);\n    }\n  }, [loading]);\n\n  const onToggleOpen = (id: string) => {\n    setOpenFolders((prev) => {\n      if (prev.has(id)) {\n        prev.delete(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      } else {\n        prev.add(id);\n      }\n      return new Set(prev);\n    });\n  };\n\n  const onCreateNewFolder = (props: { idPaths: number[]; parentId?: string }) => {\n    setNewFolder(props);\n  };\n\n  const onSaveNewFolder = async (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      const newFolderId = await onNewFolder({ ...newFolder, value });\n      if (newFolderId) {\n        setMarkedFolderId(newFolderId);\n        setFocusedFolderId(newFolderId);\n        // Open current folder in case it was closed..\n        setOpenFolders((prev) => {\n          if (newFolder.parentId) {\n            prev.add(newFolder.parentId);\n          }\n          return new Set(prev);\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        />\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"]} */"));
54
54
 
55
55
  var TreeStructure = function TreeStructure(_ref) {
56
56
  var data = _ref.data,
@@ -190,7 +190,6 @@ var TreeStructure = function TreeStructure(_ref) {
190
190
  setFocusedFolderId(id);
191
191
  };
192
192
 
193
- var disableAddFolderButton = markedFolderId === undefined || getPathOfFolder(data, markedFolderId).length >= MAX_LEVEL_FOR_FOLDERS;
194
193
  return ___EmotionJSX("div", {
195
194
  ref: treestructureRef,
196
195
  onKeyDown: function onKeyDown(e) {
@@ -236,10 +235,8 @@ var TreeStructure = function TreeStructure(_ref) {
236
235
  tooltip: t('myNdla.newFolderUnder', {
237
236
  folderName: getFolderName(data, markedFolderId)
238
237
  })
239
- }, ___EmotionJSX(Button, {
240
- size: "small",
241
- light: true,
242
- disabled: disableAddFolderButton,
238
+ }, ___EmotionJSX(AddButton, {
239
+ "aria-label": t('myNdla.newFolder'),
243
240
  onClick: function onClick() {
244
241
  var paths = getPathOfFolder(data, markedFolderId || '');
245
242
  var idPaths = getIdPathsOfFolder(data, markedFolderId || '');
@@ -248,7 +245,7 @@ var TreeStructure = function TreeStructure(_ref) {
248
245
  parentId: paths[paths.length - 1]
249
246
  });
250
247
  }
251
- }, t('myNdla.newFolder')))));
248
+ }))));
252
249
  };
253
250
 
254
251
  export default TreeStructure;