@ndla/ui 56.0.13-alpha.0 → 56.0.15-alpha.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (74) hide show
  1. package/dist/panda.buildinfo.json +7 -2
  2. package/dist/styles.css +24 -4
  3. package/es/Article/Article.js +4 -1
  4. package/es/ContentTypeBadge/ContentTypeBadge.js +2 -0
  5. package/es/TreeStructure/TreeStructure.js +292 -181
  6. package/es/TreeStructure/helperFunctions.js +0 -3
  7. package/es/TreeStructure/index.js +1 -2
  8. package/es/index.js +0 -1
  9. package/es/locale/messages-en.js +1 -1
  10. package/es/locale/messages-nb.js +1 -1
  11. package/es/locale/messages-nn.js +1 -1
  12. package/es/locale/messages-se.js +1 -1
  13. package/es/locale/messages-sma.js +1 -1
  14. package/es/styles.css +24 -4
  15. package/lib/Article/Article.js +4 -1
  16. package/lib/ContentTypeBadge/ContentTypeBadge.js +2 -0
  17. package/lib/TreeStructure/TreeStructure.d.ts +7 -6
  18. package/lib/TreeStructure/TreeStructure.js +293 -180
  19. package/lib/TreeStructure/helperFunctions.d.ts +0 -2
  20. package/lib/TreeStructure/helperFunctions.js +2 -6
  21. package/lib/TreeStructure/index.d.ts +1 -2
  22. package/lib/TreeStructure/index.js +2 -3
  23. package/lib/TreeStructure/types.d.ts +4 -22
  24. package/lib/index.d.ts +0 -2
  25. package/lib/index.js +0 -7
  26. package/lib/locale/messages-en.js +1 -1
  27. package/lib/locale/messages-nb.js +1 -1
  28. package/lib/locale/messages-nn.js +1 -1
  29. package/lib/locale/messages-se.js +1 -1
  30. package/lib/locale/messages-sma.js +1 -1
  31. package/lib/styles.css +24 -4
  32. package/package.json +7 -8
  33. package/src/Article/Article.tsx +4 -1
  34. package/src/ContentTypeBadge/ContentTypeBadge.tsx +2 -0
  35. package/src/TreeStructure/TreeStructure.stories.tsx +38 -68
  36. package/src/TreeStructure/TreeStructure.tsx +307 -194
  37. package/src/TreeStructure/helperFunctions.ts +0 -5
  38. package/src/TreeStructure/index.ts +1 -2
  39. package/src/TreeStructure/types.ts +4 -25
  40. package/src/index.ts +0 -3
  41. package/src/locale/messages-en.ts +1 -1
  42. package/src/locale/messages-nb.ts +1 -1
  43. package/src/locale/messages-nn.ts +1 -1
  44. package/src/locale/messages-se.ts +1 -1
  45. package/src/locale/messages-sma.ts +1 -1
  46. package/es/ProgrammeCard/ProgrammeCard.js +0 -51
  47. package/es/ProgrammeCard/index.js +0 -9
  48. package/es/TreeStructure/AddFolderButton.js +0 -80
  49. package/es/TreeStructure/ComboboxButton.js +0 -127
  50. package/es/TreeStructure/FolderItem.js +0 -225
  51. package/es/TreeStructure/FolderItems.js +0 -95
  52. package/es/TreeStructure/arrowNavigation.js +0 -35
  53. package/lib/ProgrammeCard/ProgrammeCard.d.ts +0 -23
  54. package/lib/ProgrammeCard/ProgrammeCard.js +0 -58
  55. package/lib/ProgrammeCard/index.d.ts +0 -9
  56. package/lib/ProgrammeCard/index.js +0 -13
  57. package/lib/TreeStructure/AddFolderButton.d.ts +0 -17
  58. package/lib/TreeStructure/AddFolderButton.js +0 -85
  59. package/lib/TreeStructure/ComboboxButton.d.ts +0 -27
  60. package/lib/TreeStructure/ComboboxButton.js +0 -134
  61. package/lib/TreeStructure/FolderItem.d.ts +0 -17
  62. package/lib/TreeStructure/FolderItem.js +0 -230
  63. package/lib/TreeStructure/FolderItems.d.ts +0 -22
  64. package/lib/TreeStructure/FolderItems.js +0 -100
  65. package/lib/TreeStructure/arrowNavigation.d.ts +0 -10
  66. package/lib/TreeStructure/arrowNavigation.js +0 -42
  67. package/src/ProgrammeCard/ProgrammeCard.stories.tsx +0 -35
  68. package/src/ProgrammeCard/ProgrammeCard.tsx +0 -78
  69. package/src/ProgrammeCard/index.tsx +0 -10
  70. package/src/TreeStructure/AddFolderButton.tsx +0 -79
  71. package/src/TreeStructure/ComboboxButton.tsx +0 -172
  72. package/src/TreeStructure/FolderItem.tsx +0 -307
  73. package/src/TreeStructure/FolderItems.tsx +0 -121
  74. package/src/TreeStructure/arrowNavigation.ts +0 -54
@@ -1,225 +0,0 @@
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
- /**
4
- * Copyright (c) 2022-present, NDLA.
5
- *
6
- * This source code is licensed under the GPLv3 license found in the
7
- * LICENSE file in the root directory of this source tree.
8
- *
9
- */
10
-
11
- import { useEffect, useMemo, useRef } from "react";
12
- import { useTranslation } from "react-i18next";
13
- import { ButtonV2 as Button } from "@ndla/button";
14
- import { colors, spacing, animations, misc, fonts } from "@ndla/core";
15
- import { ArrowDownShortLine } from "@ndla/icons/common";
16
- import { FolderUserFill } from "@ndla/icons/contentType";
17
- import { CheckLine, FolderLine } from "@ndla/icons/editor";
18
- import { SafeLink } from "@ndla/safelink";
19
- import { arrowNavigation } from "./arrowNavigation";
20
- import { treestructureId } from "./helperFunctions";
21
- import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
22
- const OpenButton = /*#__PURE__*/_styled("span", {
23
- target: "e11ok6h86",
24
- label: "OpenButton"
25
- })("display:flex;align-items:center;justify-content:center;align-self:stretch;color:", colors.brand.tertiary, ";", misc.transition.default, ";cursor:pointer;&:hover{color:", colors.brand.primary, ";}svg{width:24px;height:24px;transform:rotate(-90deg);}&[data-open=\"true\"]{svg{transform:rotate(0deg);}}&[data-hide-arrow=\"true\"]{visibility:hidden;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FolderItem.tsx"],"names":[],"mappings":"AAsB8B","file":"FolderItem.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 { CSSProperties, KeyboardEvent, useEffect, useMemo, useRef } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { ButtonV2 as Button } from \"@ndla/button\";\nimport { colors, spacing, animations, misc, fonts } from \"@ndla/core\";\nimport { ArrowDownShortLine } from \"@ndla/icons/common\";\nimport { FolderUserFill } from \"@ndla/icons/contentType\";\nimport { CheckLine, FolderLine } from \"@ndla/icons/editor\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport { arrowNavigation } from \"./arrowNavigation\";\nimport { treestructureId } from \"./helperFunctions\";\nimport { CommonFolderItemsProps } from \"./types\";\n\nconst OpenButton = styled.span`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  align-self: stretch;\n  color: ${colors.brand.tertiary};\n  ${misc.transition.default};\n  cursor: pointer;\n  &:hover {\n    color: ${colors.brand.primary};\n  }\n  svg {\n    width: 24px;\n    height: 24px;\n    transform: rotate(-90deg);\n  }\n  &[data-open=\"true\"] {\n    svg {\n      transform: rotate(0deg);\n    }\n  }\n  &[data-hide-arrow=\"true\"] {\n    visibility: hidden;\n  }\n`;\n\nconst StyledName = styled.span`\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  grid-column-start: 2;\n  text-align: left;\n`;\n\nconst IconWrapper = styled.div`\n  display: flex;\n`;\n\nconst FolderIconWrapper = styled.div`\n  svg {\n    height: 24px;\n    width: 24px;\n  }\n`;\n\nconst FolderName = styled(Button)`\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  border: none;\n  outline: none;\n  color: ${colors.text.primary};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n\n  &:hover {\n    box-shadow: none;\n    outline: none;\n    background: ${colors.brand.lightest};\n    color: ${colors.text.primary};\n  }\n\n  &[data-focused=\"true\"] {\n    background: ${colors.brand.lightest};\n  }\n\n  &[data-selected=\"true\"] {\n    background: ${colors.brand.lighter};\n    &:hover {\n      background: ${colors.brand.light};\n    }\n  }\n\n  &[data-creating=\"true\"][data-focused=\"true\"] {\n    color: ${colors.brand.primary};\n  }\n\n  &[data-creating=\"true\"] {\n    background: none;\n  }\n`;\n\nconst StyledCheck = styled(CheckLine)`\n  color: ${colors.support.green};\n`;\n\nconst FolderNameLink = styled(SafeLink)`\n  display: grid;\n  align-items: center;\n  grid-template-columns: ${spacing.medium} 1fr auto;\n  padding: ${spacing.small} ${spacing.xxsmall};\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  cursor: pointer;\n\n  border: none;\n  box-shadow: none;\n  color: ${colors.text.primary};\n  ${fonts.sizes(\"16px\")};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n  &[data-selected=\"true\"] {\n    color: ${colors.brand.primary};\n    font-weight: ${fonts.weight.semibold};\n  }\n  &:hover,\n  &:focus {\n    color: ${colors.brand.primary};\n  }\n`;\n\ninterface Props extends CommonFolderItemsProps {\n  isOpen: boolean;\n  folder: IFolder;\n  isCreatingFolder?: boolean;\n  index: number;\n}\n\nconst FolderItem = ({\n  focusedFolder,\n  folder,\n  isOpen,\n  level,\n  loading,\n  selectedFolder,\n  onCloseFolder,\n  onOpenFolder,\n  setFocusedFolder,\n  setSelectedFolder,\n  targetResource,\n  visibleFolders,\n  maxLevel,\n  isCreatingFolder,\n  type,\n  closeTree,\n  index,\n}: Props) => {\n  const { t } = useTranslation();\n  const { id, name } = folder;\n  const ref = useRef<HTMLButtonElement & HTMLAnchorElement>(null);\n  const selected = selectedFolder ? selectedFolder.id === id : false;\n\n  const levelVariable = useMemo(() => ({ \"--level\": level }) as unknown as CSSProperties, [level]);\n\n  const focused = focusedFolder?.id === id;\n\n  const handleClickFolder = () => {\n    if (!selected) {\n      setSelectedFolder(folder);\n    }\n    setFocusedFolder(folder);\n    if (type === \"picker\") {\n      if (selected) {\n        closeTree();\n      }\n    }\n  };\n\n  useEffect(() => {\n    if (focusedFolder?.id === id && !isCreatingFolder) {\n      if (type === \"navigation\") {\n        ref.current?.focus();\n      }\n      if (type === \"picker\") {\n        ref.current?.scrollIntoView({\n          behavior: \"smooth\",\n          block: \"start\",\n        });\n      }\n    }\n  }, [focusedFolder, ref, id, isCreatingFolder, type]);\n\n  const linkPath = `/minndla/folders/${id}`;\n\n  const containsResource =\n    targetResource && folder.resources.some((resource) => resource.resourceId === targetResource.resourceId);\n\n  const emptyFolder = folder.subfolders.length === 0;\n  const isMaxDepth = level > maxLevel;\n  const hideArrow = isMaxDepth || emptyFolder;\n\n  const FolderIcon = folder.status === \"shared\" ? FolderUserFill : FolderLine;\n\n  const tabable = selected || focused || (!focusedFolder && !folder.parentId && index === 0);\n\n  return type === \"navigation\" ? (\n    <FolderNameLink\n      role=\"treeitem\"\n      aria-owns={folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-current={selected ? \"page\" : undefined}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      ref={ref}\n      style={levelVariable}\n      onKeyDown={(e: KeyboardEvent<HTMLElement>) => {\n        if (e.key === \"Enter\") {\n          setSelectedFolder(folder);\n          return;\n        }\n        arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);\n      }}\n      to={loading ? \"\" : linkPath}\n      tabIndex={tabable ? 0 : -1}\n      data-selected={selected}\n      onFocus={() => setFocusedFolder(folder)}\n      onClick={handleClickFolder}\n    >\n      <OpenButton\n        aria-hidden\n        tabIndex={-1}\n        data-open={isOpen}\n        data-hide-arrow={hideArrow}\n        onClick={(e) => {\n          e.stopPropagation();\n          e.preventDefault();\n          ref.current?.focus();\n          if (isOpen) {\n            onCloseFolder(id);\n          } else {\n            onOpenFolder(id);\n          }\n        }}\n      >\n        <ArrowDownShortLine />\n      </OpenButton>\n      <StyledName>{name}</StyledName>\n    </FolderNameLink>\n  ) : (\n    <FolderName\n      tabIndex={-1}\n      role=\"treeitem\"\n      id={treestructureId(type, folder.id)}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-selected={selected}\n      data-focused={focusedFolder?.id === folder.id}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      aria-label={`${name}${folder.status === \"shared\" ? `, ${t(\"myNdla.folder.sharing.shared\")}` : \"\"}`}\n      variant=\"ghost\"\n      shape=\"sharp\"\n      fontWeight=\"normal\"\n      colorTheme=\"light\"\n      ref={ref}\n      style={levelVariable}\n      data-selected={selected}\n      disabled={loading}\n      onFocus={() => setFocusedFolder(focusedFolder || folder)}\n      onClick={handleClickFolder}\n      data-creating={isCreatingFolder}\n    >\n      <IconWrapper>\n        <OpenButton\n          aria-hidden\n          tabIndex={-1}\n          data-open={isOpen}\n          data-hide-arrow={hideArrow}\n          onClick={(e) => {\n            e.stopPropagation();\n            setFocusedFolder(folder);\n            if (isOpen) {\n              onCloseFolder(id);\n            } else {\n              onOpenFolder(id);\n            }\n          }}\n        >\n          <ArrowDownShortLine />\n        </OpenButton>\n        <FolderIconWrapper>\n          <FolderIcon />\n        </FolderIconWrapper>\n      </IconWrapper>\n      <StyledName>{name}</StyledName>\n      {containsResource && (\n        <StyledCheck\n          aria-label={t(\"myNdla.alreadyInFolder\")}\n          id={`alreadyAdded-${folder.id}`}\n          title={t(\"myNdla.alreadyInFolder\")}\n        />\n      )}\n    </FolderName>\n  );\n};\n\nexport default FolderItem;\n"]} */"));
26
- const StyledName = /*#__PURE__*/_styled("span", {
27
- target: "e11ok6h85",
28
- label: "StyledName"
29
- })(process.env.NODE_ENV === "production" ? {
30
- name: "woa3z4",
31
- styles: "white-space:nowrap;overflow:hidden;text-overflow:ellipsis;grid-column-start:2;text-align:left"
32
- } : {
33
- name: "woa3z4",
34
- styles: "white-space:nowrap;overflow:hidden;text-overflow:ellipsis;grid-column-start:2;text-align:left",
35
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FolderItem.tsx"],"names":[],"mappings":"AAgD8B","file":"FolderItem.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 { CSSProperties, KeyboardEvent, useEffect, useMemo, useRef } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { ButtonV2 as Button } from \"@ndla/button\";\nimport { colors, spacing, animations, misc, fonts } from \"@ndla/core\";\nimport { ArrowDownShortLine } from \"@ndla/icons/common\";\nimport { FolderUserFill } from \"@ndla/icons/contentType\";\nimport { CheckLine, FolderLine } from \"@ndla/icons/editor\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport { arrowNavigation } from \"./arrowNavigation\";\nimport { treestructureId } from \"./helperFunctions\";\nimport { CommonFolderItemsProps } from \"./types\";\n\nconst OpenButton = styled.span`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  align-self: stretch;\n  color: ${colors.brand.tertiary};\n  ${misc.transition.default};\n  cursor: pointer;\n  &:hover {\n    color: ${colors.brand.primary};\n  }\n  svg {\n    width: 24px;\n    height: 24px;\n    transform: rotate(-90deg);\n  }\n  &[data-open=\"true\"] {\n    svg {\n      transform: rotate(0deg);\n    }\n  }\n  &[data-hide-arrow=\"true\"] {\n    visibility: hidden;\n  }\n`;\n\nconst StyledName = styled.span`\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  grid-column-start: 2;\n  text-align: left;\n`;\n\nconst IconWrapper = styled.div`\n  display: flex;\n`;\n\nconst FolderIconWrapper = styled.div`\n  svg {\n    height: 24px;\n    width: 24px;\n  }\n`;\n\nconst FolderName = styled(Button)`\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  border: none;\n  outline: none;\n  color: ${colors.text.primary};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n\n  &:hover {\n    box-shadow: none;\n    outline: none;\n    background: ${colors.brand.lightest};\n    color: ${colors.text.primary};\n  }\n\n  &[data-focused=\"true\"] {\n    background: ${colors.brand.lightest};\n  }\n\n  &[data-selected=\"true\"] {\n    background: ${colors.brand.lighter};\n    &:hover {\n      background: ${colors.brand.light};\n    }\n  }\n\n  &[data-creating=\"true\"][data-focused=\"true\"] {\n    color: ${colors.brand.primary};\n  }\n\n  &[data-creating=\"true\"] {\n    background: none;\n  }\n`;\n\nconst StyledCheck = styled(CheckLine)`\n  color: ${colors.support.green};\n`;\n\nconst FolderNameLink = styled(SafeLink)`\n  display: grid;\n  align-items: center;\n  grid-template-columns: ${spacing.medium} 1fr auto;\n  padding: ${spacing.small} ${spacing.xxsmall};\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  cursor: pointer;\n\n  border: none;\n  box-shadow: none;\n  color: ${colors.text.primary};\n  ${fonts.sizes(\"16px\")};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n  &[data-selected=\"true\"] {\n    color: ${colors.brand.primary};\n    font-weight: ${fonts.weight.semibold};\n  }\n  &:hover,\n  &:focus {\n    color: ${colors.brand.primary};\n  }\n`;\n\ninterface Props extends CommonFolderItemsProps {\n  isOpen: boolean;\n  folder: IFolder;\n  isCreatingFolder?: boolean;\n  index: number;\n}\n\nconst FolderItem = ({\n  focusedFolder,\n  folder,\n  isOpen,\n  level,\n  loading,\n  selectedFolder,\n  onCloseFolder,\n  onOpenFolder,\n  setFocusedFolder,\n  setSelectedFolder,\n  targetResource,\n  visibleFolders,\n  maxLevel,\n  isCreatingFolder,\n  type,\n  closeTree,\n  index,\n}: Props) => {\n  const { t } = useTranslation();\n  const { id, name } = folder;\n  const ref = useRef<HTMLButtonElement & HTMLAnchorElement>(null);\n  const selected = selectedFolder ? selectedFolder.id === id : false;\n\n  const levelVariable = useMemo(() => ({ \"--level\": level }) as unknown as CSSProperties, [level]);\n\n  const focused = focusedFolder?.id === id;\n\n  const handleClickFolder = () => {\n    if (!selected) {\n      setSelectedFolder(folder);\n    }\n    setFocusedFolder(folder);\n    if (type === \"picker\") {\n      if (selected) {\n        closeTree();\n      }\n    }\n  };\n\n  useEffect(() => {\n    if (focusedFolder?.id === id && !isCreatingFolder) {\n      if (type === \"navigation\") {\n        ref.current?.focus();\n      }\n      if (type === \"picker\") {\n        ref.current?.scrollIntoView({\n          behavior: \"smooth\",\n          block: \"start\",\n        });\n      }\n    }\n  }, [focusedFolder, ref, id, isCreatingFolder, type]);\n\n  const linkPath = `/minndla/folders/${id}`;\n\n  const containsResource =\n    targetResource && folder.resources.some((resource) => resource.resourceId === targetResource.resourceId);\n\n  const emptyFolder = folder.subfolders.length === 0;\n  const isMaxDepth = level > maxLevel;\n  const hideArrow = isMaxDepth || emptyFolder;\n\n  const FolderIcon = folder.status === \"shared\" ? FolderUserFill : FolderLine;\n\n  const tabable = selected || focused || (!focusedFolder && !folder.parentId && index === 0);\n\n  return type === \"navigation\" ? (\n    <FolderNameLink\n      role=\"treeitem\"\n      aria-owns={folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-current={selected ? \"page\" : undefined}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      ref={ref}\n      style={levelVariable}\n      onKeyDown={(e: KeyboardEvent<HTMLElement>) => {\n        if (e.key === \"Enter\") {\n          setSelectedFolder(folder);\n          return;\n        }\n        arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);\n      }}\n      to={loading ? \"\" : linkPath}\n      tabIndex={tabable ? 0 : -1}\n      data-selected={selected}\n      onFocus={() => setFocusedFolder(folder)}\n      onClick={handleClickFolder}\n    >\n      <OpenButton\n        aria-hidden\n        tabIndex={-1}\n        data-open={isOpen}\n        data-hide-arrow={hideArrow}\n        onClick={(e) => {\n          e.stopPropagation();\n          e.preventDefault();\n          ref.current?.focus();\n          if (isOpen) {\n            onCloseFolder(id);\n          } else {\n            onOpenFolder(id);\n          }\n        }}\n      >\n        <ArrowDownShortLine />\n      </OpenButton>\n      <StyledName>{name}</StyledName>\n    </FolderNameLink>\n  ) : (\n    <FolderName\n      tabIndex={-1}\n      role=\"treeitem\"\n      id={treestructureId(type, folder.id)}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-selected={selected}\n      data-focused={focusedFolder?.id === folder.id}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      aria-label={`${name}${folder.status === \"shared\" ? `, ${t(\"myNdla.folder.sharing.shared\")}` : \"\"}`}\n      variant=\"ghost\"\n      shape=\"sharp\"\n      fontWeight=\"normal\"\n      colorTheme=\"light\"\n      ref={ref}\n      style={levelVariable}\n      data-selected={selected}\n      disabled={loading}\n      onFocus={() => setFocusedFolder(focusedFolder || folder)}\n      onClick={handleClickFolder}\n      data-creating={isCreatingFolder}\n    >\n      <IconWrapper>\n        <OpenButton\n          aria-hidden\n          tabIndex={-1}\n          data-open={isOpen}\n          data-hide-arrow={hideArrow}\n          onClick={(e) => {\n            e.stopPropagation();\n            setFocusedFolder(folder);\n            if (isOpen) {\n              onCloseFolder(id);\n            } else {\n              onOpenFolder(id);\n            }\n          }}\n        >\n          <ArrowDownShortLine />\n        </OpenButton>\n        <FolderIconWrapper>\n          <FolderIcon />\n        </FolderIconWrapper>\n      </IconWrapper>\n      <StyledName>{name}</StyledName>\n      {containsResource && (\n        <StyledCheck\n          aria-label={t(\"myNdla.alreadyInFolder\")}\n          id={`alreadyAdded-${folder.id}`}\n          title={t(\"myNdla.alreadyInFolder\")}\n        />\n      )}\n    </FolderName>\n  );\n};\n\nexport default FolderItem;\n"]} */",
36
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
37
- });
38
- const IconWrapper = /*#__PURE__*/_styled("div", {
39
- target: "e11ok6h84",
40
- label: "IconWrapper"
41
- })(process.env.NODE_ENV === "production" ? {
42
- name: "zjik7",
43
- styles: "display:flex"
44
- } : {
45
- name: "zjik7",
46
- styles: "display:flex",
47
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FolderItem.tsx"],"names":[],"mappings":"AAwD8B","file":"FolderItem.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 { CSSProperties, KeyboardEvent, useEffect, useMemo, useRef } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { ButtonV2 as Button } from \"@ndla/button\";\nimport { colors, spacing, animations, misc, fonts } from \"@ndla/core\";\nimport { ArrowDownShortLine } from \"@ndla/icons/common\";\nimport { FolderUserFill } from \"@ndla/icons/contentType\";\nimport { CheckLine, FolderLine } from \"@ndla/icons/editor\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport { arrowNavigation } from \"./arrowNavigation\";\nimport { treestructureId } from \"./helperFunctions\";\nimport { CommonFolderItemsProps } from \"./types\";\n\nconst OpenButton = styled.span`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  align-self: stretch;\n  color: ${colors.brand.tertiary};\n  ${misc.transition.default};\n  cursor: pointer;\n  &:hover {\n    color: ${colors.brand.primary};\n  }\n  svg {\n    width: 24px;\n    height: 24px;\n    transform: rotate(-90deg);\n  }\n  &[data-open=\"true\"] {\n    svg {\n      transform: rotate(0deg);\n    }\n  }\n  &[data-hide-arrow=\"true\"] {\n    visibility: hidden;\n  }\n`;\n\nconst StyledName = styled.span`\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  grid-column-start: 2;\n  text-align: left;\n`;\n\nconst IconWrapper = styled.div`\n  display: flex;\n`;\n\nconst FolderIconWrapper = styled.div`\n  svg {\n    height: 24px;\n    width: 24px;\n  }\n`;\n\nconst FolderName = styled(Button)`\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  border: none;\n  outline: none;\n  color: ${colors.text.primary};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n\n  &:hover {\n    box-shadow: none;\n    outline: none;\n    background: ${colors.brand.lightest};\n    color: ${colors.text.primary};\n  }\n\n  &[data-focused=\"true\"] {\n    background: ${colors.brand.lightest};\n  }\n\n  &[data-selected=\"true\"] {\n    background: ${colors.brand.lighter};\n    &:hover {\n      background: ${colors.brand.light};\n    }\n  }\n\n  &[data-creating=\"true\"][data-focused=\"true\"] {\n    color: ${colors.brand.primary};\n  }\n\n  &[data-creating=\"true\"] {\n    background: none;\n  }\n`;\n\nconst StyledCheck = styled(CheckLine)`\n  color: ${colors.support.green};\n`;\n\nconst FolderNameLink = styled(SafeLink)`\n  display: grid;\n  align-items: center;\n  grid-template-columns: ${spacing.medium} 1fr auto;\n  padding: ${spacing.small} ${spacing.xxsmall};\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  cursor: pointer;\n\n  border: none;\n  box-shadow: none;\n  color: ${colors.text.primary};\n  ${fonts.sizes(\"16px\")};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n  &[data-selected=\"true\"] {\n    color: ${colors.brand.primary};\n    font-weight: ${fonts.weight.semibold};\n  }\n  &:hover,\n  &:focus {\n    color: ${colors.brand.primary};\n  }\n`;\n\ninterface Props extends CommonFolderItemsProps {\n  isOpen: boolean;\n  folder: IFolder;\n  isCreatingFolder?: boolean;\n  index: number;\n}\n\nconst FolderItem = ({\n  focusedFolder,\n  folder,\n  isOpen,\n  level,\n  loading,\n  selectedFolder,\n  onCloseFolder,\n  onOpenFolder,\n  setFocusedFolder,\n  setSelectedFolder,\n  targetResource,\n  visibleFolders,\n  maxLevel,\n  isCreatingFolder,\n  type,\n  closeTree,\n  index,\n}: Props) => {\n  const { t } = useTranslation();\n  const { id, name } = folder;\n  const ref = useRef<HTMLButtonElement & HTMLAnchorElement>(null);\n  const selected = selectedFolder ? selectedFolder.id === id : false;\n\n  const levelVariable = useMemo(() => ({ \"--level\": level }) as unknown as CSSProperties, [level]);\n\n  const focused = focusedFolder?.id === id;\n\n  const handleClickFolder = () => {\n    if (!selected) {\n      setSelectedFolder(folder);\n    }\n    setFocusedFolder(folder);\n    if (type === \"picker\") {\n      if (selected) {\n        closeTree();\n      }\n    }\n  };\n\n  useEffect(() => {\n    if (focusedFolder?.id === id && !isCreatingFolder) {\n      if (type === \"navigation\") {\n        ref.current?.focus();\n      }\n      if (type === \"picker\") {\n        ref.current?.scrollIntoView({\n          behavior: \"smooth\",\n          block: \"start\",\n        });\n      }\n    }\n  }, [focusedFolder, ref, id, isCreatingFolder, type]);\n\n  const linkPath = `/minndla/folders/${id}`;\n\n  const containsResource =\n    targetResource && folder.resources.some((resource) => resource.resourceId === targetResource.resourceId);\n\n  const emptyFolder = folder.subfolders.length === 0;\n  const isMaxDepth = level > maxLevel;\n  const hideArrow = isMaxDepth || emptyFolder;\n\n  const FolderIcon = folder.status === \"shared\" ? FolderUserFill : FolderLine;\n\n  const tabable = selected || focused || (!focusedFolder && !folder.parentId && index === 0);\n\n  return type === \"navigation\" ? (\n    <FolderNameLink\n      role=\"treeitem\"\n      aria-owns={folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-current={selected ? \"page\" : undefined}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      ref={ref}\n      style={levelVariable}\n      onKeyDown={(e: KeyboardEvent<HTMLElement>) => {\n        if (e.key === \"Enter\") {\n          setSelectedFolder(folder);\n          return;\n        }\n        arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);\n      }}\n      to={loading ? \"\" : linkPath}\n      tabIndex={tabable ? 0 : -1}\n      data-selected={selected}\n      onFocus={() => setFocusedFolder(folder)}\n      onClick={handleClickFolder}\n    >\n      <OpenButton\n        aria-hidden\n        tabIndex={-1}\n        data-open={isOpen}\n        data-hide-arrow={hideArrow}\n        onClick={(e) => {\n          e.stopPropagation();\n          e.preventDefault();\n          ref.current?.focus();\n          if (isOpen) {\n            onCloseFolder(id);\n          } else {\n            onOpenFolder(id);\n          }\n        }}\n      >\n        <ArrowDownShortLine />\n      </OpenButton>\n      <StyledName>{name}</StyledName>\n    </FolderNameLink>\n  ) : (\n    <FolderName\n      tabIndex={-1}\n      role=\"treeitem\"\n      id={treestructureId(type, folder.id)}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-selected={selected}\n      data-focused={focusedFolder?.id === folder.id}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      aria-label={`${name}${folder.status === \"shared\" ? `, ${t(\"myNdla.folder.sharing.shared\")}` : \"\"}`}\n      variant=\"ghost\"\n      shape=\"sharp\"\n      fontWeight=\"normal\"\n      colorTheme=\"light\"\n      ref={ref}\n      style={levelVariable}\n      data-selected={selected}\n      disabled={loading}\n      onFocus={() => setFocusedFolder(focusedFolder || folder)}\n      onClick={handleClickFolder}\n      data-creating={isCreatingFolder}\n    >\n      <IconWrapper>\n        <OpenButton\n          aria-hidden\n          tabIndex={-1}\n          data-open={isOpen}\n          data-hide-arrow={hideArrow}\n          onClick={(e) => {\n            e.stopPropagation();\n            setFocusedFolder(folder);\n            if (isOpen) {\n              onCloseFolder(id);\n            } else {\n              onOpenFolder(id);\n            }\n          }}\n        >\n          <ArrowDownShortLine />\n        </OpenButton>\n        <FolderIconWrapper>\n          <FolderIcon />\n        </FolderIconWrapper>\n      </IconWrapper>\n      <StyledName>{name}</StyledName>\n      {containsResource && (\n        <StyledCheck\n          aria-label={t(\"myNdla.alreadyInFolder\")}\n          id={`alreadyAdded-${folder.id}`}\n          title={t(\"myNdla.alreadyInFolder\")}\n        />\n      )}\n    </FolderName>\n  );\n};\n\nexport default FolderItem;\n"]} */",
48
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
49
- });
50
- const FolderIconWrapper = /*#__PURE__*/_styled("div", {
51
- target: "e11ok6h83",
52
- label: "FolderIconWrapper"
53
- })(process.env.NODE_ENV === "production" ? {
54
- name: "h1kfx5",
55
- styles: "svg{height:24px;width:24px;}"
56
- } : {
57
- name: "h1kfx5",
58
- styles: "svg{height:24px;width:24px;}",
59
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FolderItem.tsx"],"names":[],"mappings":"AA4DoC","file":"FolderItem.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 { CSSProperties, KeyboardEvent, useEffect, useMemo, useRef } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { ButtonV2 as Button } from \"@ndla/button\";\nimport { colors, spacing, animations, misc, fonts } from \"@ndla/core\";\nimport { ArrowDownShortLine } from \"@ndla/icons/common\";\nimport { FolderUserFill } from \"@ndla/icons/contentType\";\nimport { CheckLine, FolderLine } from \"@ndla/icons/editor\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport { arrowNavigation } from \"./arrowNavigation\";\nimport { treestructureId } from \"./helperFunctions\";\nimport { CommonFolderItemsProps } from \"./types\";\n\nconst OpenButton = styled.span`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  align-self: stretch;\n  color: ${colors.brand.tertiary};\n  ${misc.transition.default};\n  cursor: pointer;\n  &:hover {\n    color: ${colors.brand.primary};\n  }\n  svg {\n    width: 24px;\n    height: 24px;\n    transform: rotate(-90deg);\n  }\n  &[data-open=\"true\"] {\n    svg {\n      transform: rotate(0deg);\n    }\n  }\n  &[data-hide-arrow=\"true\"] {\n    visibility: hidden;\n  }\n`;\n\nconst StyledName = styled.span`\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  grid-column-start: 2;\n  text-align: left;\n`;\n\nconst IconWrapper = styled.div`\n  display: flex;\n`;\n\nconst FolderIconWrapper = styled.div`\n  svg {\n    height: 24px;\n    width: 24px;\n  }\n`;\n\nconst FolderName = styled(Button)`\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  border: none;\n  outline: none;\n  color: ${colors.text.primary};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n\n  &:hover {\n    box-shadow: none;\n    outline: none;\n    background: ${colors.brand.lightest};\n    color: ${colors.text.primary};\n  }\n\n  &[data-focused=\"true\"] {\n    background: ${colors.brand.lightest};\n  }\n\n  &[data-selected=\"true\"] {\n    background: ${colors.brand.lighter};\n    &:hover {\n      background: ${colors.brand.light};\n    }\n  }\n\n  &[data-creating=\"true\"][data-focused=\"true\"] {\n    color: ${colors.brand.primary};\n  }\n\n  &[data-creating=\"true\"] {\n    background: none;\n  }\n`;\n\nconst StyledCheck = styled(CheckLine)`\n  color: ${colors.support.green};\n`;\n\nconst FolderNameLink = styled(SafeLink)`\n  display: grid;\n  align-items: center;\n  grid-template-columns: ${spacing.medium} 1fr auto;\n  padding: ${spacing.small} ${spacing.xxsmall};\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  cursor: pointer;\n\n  border: none;\n  box-shadow: none;\n  color: ${colors.text.primary};\n  ${fonts.sizes(\"16px\")};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n  &[data-selected=\"true\"] {\n    color: ${colors.brand.primary};\n    font-weight: ${fonts.weight.semibold};\n  }\n  &:hover,\n  &:focus {\n    color: ${colors.brand.primary};\n  }\n`;\n\ninterface Props extends CommonFolderItemsProps {\n  isOpen: boolean;\n  folder: IFolder;\n  isCreatingFolder?: boolean;\n  index: number;\n}\n\nconst FolderItem = ({\n  focusedFolder,\n  folder,\n  isOpen,\n  level,\n  loading,\n  selectedFolder,\n  onCloseFolder,\n  onOpenFolder,\n  setFocusedFolder,\n  setSelectedFolder,\n  targetResource,\n  visibleFolders,\n  maxLevel,\n  isCreatingFolder,\n  type,\n  closeTree,\n  index,\n}: Props) => {\n  const { t } = useTranslation();\n  const { id, name } = folder;\n  const ref = useRef<HTMLButtonElement & HTMLAnchorElement>(null);\n  const selected = selectedFolder ? selectedFolder.id === id : false;\n\n  const levelVariable = useMemo(() => ({ \"--level\": level }) as unknown as CSSProperties, [level]);\n\n  const focused = focusedFolder?.id === id;\n\n  const handleClickFolder = () => {\n    if (!selected) {\n      setSelectedFolder(folder);\n    }\n    setFocusedFolder(folder);\n    if (type === \"picker\") {\n      if (selected) {\n        closeTree();\n      }\n    }\n  };\n\n  useEffect(() => {\n    if (focusedFolder?.id === id && !isCreatingFolder) {\n      if (type === \"navigation\") {\n        ref.current?.focus();\n      }\n      if (type === \"picker\") {\n        ref.current?.scrollIntoView({\n          behavior: \"smooth\",\n          block: \"start\",\n        });\n      }\n    }\n  }, [focusedFolder, ref, id, isCreatingFolder, type]);\n\n  const linkPath = `/minndla/folders/${id}`;\n\n  const containsResource =\n    targetResource && folder.resources.some((resource) => resource.resourceId === targetResource.resourceId);\n\n  const emptyFolder = folder.subfolders.length === 0;\n  const isMaxDepth = level > maxLevel;\n  const hideArrow = isMaxDepth || emptyFolder;\n\n  const FolderIcon = folder.status === \"shared\" ? FolderUserFill : FolderLine;\n\n  const tabable = selected || focused || (!focusedFolder && !folder.parentId && index === 0);\n\n  return type === \"navigation\" ? (\n    <FolderNameLink\n      role=\"treeitem\"\n      aria-owns={folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-current={selected ? \"page\" : undefined}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      ref={ref}\n      style={levelVariable}\n      onKeyDown={(e: KeyboardEvent<HTMLElement>) => {\n        if (e.key === \"Enter\") {\n          setSelectedFolder(folder);\n          return;\n        }\n        arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);\n      }}\n      to={loading ? \"\" : linkPath}\n      tabIndex={tabable ? 0 : -1}\n      data-selected={selected}\n      onFocus={() => setFocusedFolder(folder)}\n      onClick={handleClickFolder}\n    >\n      <OpenButton\n        aria-hidden\n        tabIndex={-1}\n        data-open={isOpen}\n        data-hide-arrow={hideArrow}\n        onClick={(e) => {\n          e.stopPropagation();\n          e.preventDefault();\n          ref.current?.focus();\n          if (isOpen) {\n            onCloseFolder(id);\n          } else {\n            onOpenFolder(id);\n          }\n        }}\n      >\n        <ArrowDownShortLine />\n      </OpenButton>\n      <StyledName>{name}</StyledName>\n    </FolderNameLink>\n  ) : (\n    <FolderName\n      tabIndex={-1}\n      role=\"treeitem\"\n      id={treestructureId(type, folder.id)}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-selected={selected}\n      data-focused={focusedFolder?.id === folder.id}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      aria-label={`${name}${folder.status === \"shared\" ? `, ${t(\"myNdla.folder.sharing.shared\")}` : \"\"}`}\n      variant=\"ghost\"\n      shape=\"sharp\"\n      fontWeight=\"normal\"\n      colorTheme=\"light\"\n      ref={ref}\n      style={levelVariable}\n      data-selected={selected}\n      disabled={loading}\n      onFocus={() => setFocusedFolder(focusedFolder || folder)}\n      onClick={handleClickFolder}\n      data-creating={isCreatingFolder}\n    >\n      <IconWrapper>\n        <OpenButton\n          aria-hidden\n          tabIndex={-1}\n          data-open={isOpen}\n          data-hide-arrow={hideArrow}\n          onClick={(e) => {\n            e.stopPropagation();\n            setFocusedFolder(folder);\n            if (isOpen) {\n              onCloseFolder(id);\n            } else {\n              onOpenFolder(id);\n            }\n          }}\n        >\n          <ArrowDownShortLine />\n        </OpenButton>\n        <FolderIconWrapper>\n          <FolderIcon />\n        </FolderIconWrapper>\n      </IconWrapper>\n      <StyledName>{name}</StyledName>\n      {containsResource && (\n        <StyledCheck\n          aria-label={t(\"myNdla.alreadyInFolder\")}\n          id={`alreadyAdded-${folder.id}`}\n          title={t(\"myNdla.alreadyInFolder\")}\n        />\n      )}\n    </FolderName>\n  );\n};\n\nexport default FolderItem;\n"]} */",
60
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
61
- });
62
- const FolderName = /*#__PURE__*/_styled(Button, {
63
- target: "e11ok6h82",
64
- label: "FolderName"
65
- })("display:grid;grid-template-columns:auto 1fr auto;padding-left:calc(0.75 * ", spacing.normal, " * var(--level));gap:", spacing.xxsmall, ";border:none;outline:none;color:", colors.text.primary, ";transition:", animations.durations.superFast, ";word-break:break-word;&:hover{box-shadow:none;outline:none;background:", colors.brand.lightest, ";color:", colors.text.primary, ";}&[data-focused=\"true\"]{background:", colors.brand.lightest, ";}&[data-selected=\"true\"]{background:", colors.brand.lighter, ";&:hover{background:", colors.brand.light, ";}}&[data-creating=\"true\"][data-focused=\"true\"]{color:", colors.brand.primary, ";}&[data-creating=\"true\"]{background:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FolderItem.tsx"],"names":[],"mappings":"AAmEiC","file":"FolderItem.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 { CSSProperties, KeyboardEvent, useEffect, useMemo, useRef } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { ButtonV2 as Button } from \"@ndla/button\";\nimport { colors, spacing, animations, misc, fonts } from \"@ndla/core\";\nimport { ArrowDownShortLine } from \"@ndla/icons/common\";\nimport { FolderUserFill } from \"@ndla/icons/contentType\";\nimport { CheckLine, FolderLine } from \"@ndla/icons/editor\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport { arrowNavigation } from \"./arrowNavigation\";\nimport { treestructureId } from \"./helperFunctions\";\nimport { CommonFolderItemsProps } from \"./types\";\n\nconst OpenButton = styled.span`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  align-self: stretch;\n  color: ${colors.brand.tertiary};\n  ${misc.transition.default};\n  cursor: pointer;\n  &:hover {\n    color: ${colors.brand.primary};\n  }\n  svg {\n    width: 24px;\n    height: 24px;\n    transform: rotate(-90deg);\n  }\n  &[data-open=\"true\"] {\n    svg {\n      transform: rotate(0deg);\n    }\n  }\n  &[data-hide-arrow=\"true\"] {\n    visibility: hidden;\n  }\n`;\n\nconst StyledName = styled.span`\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  grid-column-start: 2;\n  text-align: left;\n`;\n\nconst IconWrapper = styled.div`\n  display: flex;\n`;\n\nconst FolderIconWrapper = styled.div`\n  svg {\n    height: 24px;\n    width: 24px;\n  }\n`;\n\nconst FolderName = styled(Button)`\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  border: none;\n  outline: none;\n  color: ${colors.text.primary};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n\n  &:hover {\n    box-shadow: none;\n    outline: none;\n    background: ${colors.brand.lightest};\n    color: ${colors.text.primary};\n  }\n\n  &[data-focused=\"true\"] {\n    background: ${colors.brand.lightest};\n  }\n\n  &[data-selected=\"true\"] {\n    background: ${colors.brand.lighter};\n    &:hover {\n      background: ${colors.brand.light};\n    }\n  }\n\n  &[data-creating=\"true\"][data-focused=\"true\"] {\n    color: ${colors.brand.primary};\n  }\n\n  &[data-creating=\"true\"] {\n    background: none;\n  }\n`;\n\nconst StyledCheck = styled(CheckLine)`\n  color: ${colors.support.green};\n`;\n\nconst FolderNameLink = styled(SafeLink)`\n  display: grid;\n  align-items: center;\n  grid-template-columns: ${spacing.medium} 1fr auto;\n  padding: ${spacing.small} ${spacing.xxsmall};\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  cursor: pointer;\n\n  border: none;\n  box-shadow: none;\n  color: ${colors.text.primary};\n  ${fonts.sizes(\"16px\")};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n  &[data-selected=\"true\"] {\n    color: ${colors.brand.primary};\n    font-weight: ${fonts.weight.semibold};\n  }\n  &:hover,\n  &:focus {\n    color: ${colors.brand.primary};\n  }\n`;\n\ninterface Props extends CommonFolderItemsProps {\n  isOpen: boolean;\n  folder: IFolder;\n  isCreatingFolder?: boolean;\n  index: number;\n}\n\nconst FolderItem = ({\n  focusedFolder,\n  folder,\n  isOpen,\n  level,\n  loading,\n  selectedFolder,\n  onCloseFolder,\n  onOpenFolder,\n  setFocusedFolder,\n  setSelectedFolder,\n  targetResource,\n  visibleFolders,\n  maxLevel,\n  isCreatingFolder,\n  type,\n  closeTree,\n  index,\n}: Props) => {\n  const { t } = useTranslation();\n  const { id, name } = folder;\n  const ref = useRef<HTMLButtonElement & HTMLAnchorElement>(null);\n  const selected = selectedFolder ? selectedFolder.id === id : false;\n\n  const levelVariable = useMemo(() => ({ \"--level\": level }) as unknown as CSSProperties, [level]);\n\n  const focused = focusedFolder?.id === id;\n\n  const handleClickFolder = () => {\n    if (!selected) {\n      setSelectedFolder(folder);\n    }\n    setFocusedFolder(folder);\n    if (type === \"picker\") {\n      if (selected) {\n        closeTree();\n      }\n    }\n  };\n\n  useEffect(() => {\n    if (focusedFolder?.id === id && !isCreatingFolder) {\n      if (type === \"navigation\") {\n        ref.current?.focus();\n      }\n      if (type === \"picker\") {\n        ref.current?.scrollIntoView({\n          behavior: \"smooth\",\n          block: \"start\",\n        });\n      }\n    }\n  }, [focusedFolder, ref, id, isCreatingFolder, type]);\n\n  const linkPath = `/minndla/folders/${id}`;\n\n  const containsResource =\n    targetResource && folder.resources.some((resource) => resource.resourceId === targetResource.resourceId);\n\n  const emptyFolder = folder.subfolders.length === 0;\n  const isMaxDepth = level > maxLevel;\n  const hideArrow = isMaxDepth || emptyFolder;\n\n  const FolderIcon = folder.status === \"shared\" ? FolderUserFill : FolderLine;\n\n  const tabable = selected || focused || (!focusedFolder && !folder.parentId && index === 0);\n\n  return type === \"navigation\" ? (\n    <FolderNameLink\n      role=\"treeitem\"\n      aria-owns={folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-current={selected ? \"page\" : undefined}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      ref={ref}\n      style={levelVariable}\n      onKeyDown={(e: KeyboardEvent<HTMLElement>) => {\n        if (e.key === \"Enter\") {\n          setSelectedFolder(folder);\n          return;\n        }\n        arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);\n      }}\n      to={loading ? \"\" : linkPath}\n      tabIndex={tabable ? 0 : -1}\n      data-selected={selected}\n      onFocus={() => setFocusedFolder(folder)}\n      onClick={handleClickFolder}\n    >\n      <OpenButton\n        aria-hidden\n        tabIndex={-1}\n        data-open={isOpen}\n        data-hide-arrow={hideArrow}\n        onClick={(e) => {\n          e.stopPropagation();\n          e.preventDefault();\n          ref.current?.focus();\n          if (isOpen) {\n            onCloseFolder(id);\n          } else {\n            onOpenFolder(id);\n          }\n        }}\n      >\n        <ArrowDownShortLine />\n      </OpenButton>\n      <StyledName>{name}</StyledName>\n    </FolderNameLink>\n  ) : (\n    <FolderName\n      tabIndex={-1}\n      role=\"treeitem\"\n      id={treestructureId(type, folder.id)}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-selected={selected}\n      data-focused={focusedFolder?.id === folder.id}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      aria-label={`${name}${folder.status === \"shared\" ? `, ${t(\"myNdla.folder.sharing.shared\")}` : \"\"}`}\n      variant=\"ghost\"\n      shape=\"sharp\"\n      fontWeight=\"normal\"\n      colorTheme=\"light\"\n      ref={ref}\n      style={levelVariable}\n      data-selected={selected}\n      disabled={loading}\n      onFocus={() => setFocusedFolder(focusedFolder || folder)}\n      onClick={handleClickFolder}\n      data-creating={isCreatingFolder}\n    >\n      <IconWrapper>\n        <OpenButton\n          aria-hidden\n          tabIndex={-1}\n          data-open={isOpen}\n          data-hide-arrow={hideArrow}\n          onClick={(e) => {\n            e.stopPropagation();\n            setFocusedFolder(folder);\n            if (isOpen) {\n              onCloseFolder(id);\n            } else {\n              onOpenFolder(id);\n            }\n          }}\n        >\n          <ArrowDownShortLine />\n        </OpenButton>\n        <FolderIconWrapper>\n          <FolderIcon />\n        </FolderIconWrapper>\n      </IconWrapper>\n      <StyledName>{name}</StyledName>\n      {containsResource && (\n        <StyledCheck\n          aria-label={t(\"myNdla.alreadyInFolder\")}\n          id={`alreadyAdded-${folder.id}`}\n          title={t(\"myNdla.alreadyInFolder\")}\n        />\n      )}\n    </FolderName>\n  );\n};\n\nexport default FolderItem;\n"]} */"));
66
- const StyledCheck = /*#__PURE__*/_styled(CheckLine, {
67
- target: "e11ok6h81",
68
- label: "StyledCheck"
69
- })("color:", colors.support.green, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FolderItem.tsx"],"names":[],"mappings":"AAyGqC","file":"FolderItem.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 { CSSProperties, KeyboardEvent, useEffect, useMemo, useRef } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { ButtonV2 as Button } from \"@ndla/button\";\nimport { colors, spacing, animations, misc, fonts } from \"@ndla/core\";\nimport { ArrowDownShortLine } from \"@ndla/icons/common\";\nimport { FolderUserFill } from \"@ndla/icons/contentType\";\nimport { CheckLine, FolderLine } from \"@ndla/icons/editor\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport { arrowNavigation } from \"./arrowNavigation\";\nimport { treestructureId } from \"./helperFunctions\";\nimport { CommonFolderItemsProps } from \"./types\";\n\nconst OpenButton = styled.span`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  align-self: stretch;\n  color: ${colors.brand.tertiary};\n  ${misc.transition.default};\n  cursor: pointer;\n  &:hover {\n    color: ${colors.brand.primary};\n  }\n  svg {\n    width: 24px;\n    height: 24px;\n    transform: rotate(-90deg);\n  }\n  &[data-open=\"true\"] {\n    svg {\n      transform: rotate(0deg);\n    }\n  }\n  &[data-hide-arrow=\"true\"] {\n    visibility: hidden;\n  }\n`;\n\nconst StyledName = styled.span`\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  grid-column-start: 2;\n  text-align: left;\n`;\n\nconst IconWrapper = styled.div`\n  display: flex;\n`;\n\nconst FolderIconWrapper = styled.div`\n  svg {\n    height: 24px;\n    width: 24px;\n  }\n`;\n\nconst FolderName = styled(Button)`\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  border: none;\n  outline: none;\n  color: ${colors.text.primary};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n\n  &:hover {\n    box-shadow: none;\n    outline: none;\n    background: ${colors.brand.lightest};\n    color: ${colors.text.primary};\n  }\n\n  &[data-focused=\"true\"] {\n    background: ${colors.brand.lightest};\n  }\n\n  &[data-selected=\"true\"] {\n    background: ${colors.brand.lighter};\n    &:hover {\n      background: ${colors.brand.light};\n    }\n  }\n\n  &[data-creating=\"true\"][data-focused=\"true\"] {\n    color: ${colors.brand.primary};\n  }\n\n  &[data-creating=\"true\"] {\n    background: none;\n  }\n`;\n\nconst StyledCheck = styled(CheckLine)`\n  color: ${colors.support.green};\n`;\n\nconst FolderNameLink = styled(SafeLink)`\n  display: grid;\n  align-items: center;\n  grid-template-columns: ${spacing.medium} 1fr auto;\n  padding: ${spacing.small} ${spacing.xxsmall};\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  cursor: pointer;\n\n  border: none;\n  box-shadow: none;\n  color: ${colors.text.primary};\n  ${fonts.sizes(\"16px\")};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n  &[data-selected=\"true\"] {\n    color: ${colors.brand.primary};\n    font-weight: ${fonts.weight.semibold};\n  }\n  &:hover,\n  &:focus {\n    color: ${colors.brand.primary};\n  }\n`;\n\ninterface Props extends CommonFolderItemsProps {\n  isOpen: boolean;\n  folder: IFolder;\n  isCreatingFolder?: boolean;\n  index: number;\n}\n\nconst FolderItem = ({\n  focusedFolder,\n  folder,\n  isOpen,\n  level,\n  loading,\n  selectedFolder,\n  onCloseFolder,\n  onOpenFolder,\n  setFocusedFolder,\n  setSelectedFolder,\n  targetResource,\n  visibleFolders,\n  maxLevel,\n  isCreatingFolder,\n  type,\n  closeTree,\n  index,\n}: Props) => {\n  const { t } = useTranslation();\n  const { id, name } = folder;\n  const ref = useRef<HTMLButtonElement & HTMLAnchorElement>(null);\n  const selected = selectedFolder ? selectedFolder.id === id : false;\n\n  const levelVariable = useMemo(() => ({ \"--level\": level }) as unknown as CSSProperties, [level]);\n\n  const focused = focusedFolder?.id === id;\n\n  const handleClickFolder = () => {\n    if (!selected) {\n      setSelectedFolder(folder);\n    }\n    setFocusedFolder(folder);\n    if (type === \"picker\") {\n      if (selected) {\n        closeTree();\n      }\n    }\n  };\n\n  useEffect(() => {\n    if (focusedFolder?.id === id && !isCreatingFolder) {\n      if (type === \"navigation\") {\n        ref.current?.focus();\n      }\n      if (type === \"picker\") {\n        ref.current?.scrollIntoView({\n          behavior: \"smooth\",\n          block: \"start\",\n        });\n      }\n    }\n  }, [focusedFolder, ref, id, isCreatingFolder, type]);\n\n  const linkPath = `/minndla/folders/${id}`;\n\n  const containsResource =\n    targetResource && folder.resources.some((resource) => resource.resourceId === targetResource.resourceId);\n\n  const emptyFolder = folder.subfolders.length === 0;\n  const isMaxDepth = level > maxLevel;\n  const hideArrow = isMaxDepth || emptyFolder;\n\n  const FolderIcon = folder.status === \"shared\" ? FolderUserFill : FolderLine;\n\n  const tabable = selected || focused || (!focusedFolder && !folder.parentId && index === 0);\n\n  return type === \"navigation\" ? (\n    <FolderNameLink\n      role=\"treeitem\"\n      aria-owns={folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-current={selected ? \"page\" : undefined}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      ref={ref}\n      style={levelVariable}\n      onKeyDown={(e: KeyboardEvent<HTMLElement>) => {\n        if (e.key === \"Enter\") {\n          setSelectedFolder(folder);\n          return;\n        }\n        arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);\n      }}\n      to={loading ? \"\" : linkPath}\n      tabIndex={tabable ? 0 : -1}\n      data-selected={selected}\n      onFocus={() => setFocusedFolder(folder)}\n      onClick={handleClickFolder}\n    >\n      <OpenButton\n        aria-hidden\n        tabIndex={-1}\n        data-open={isOpen}\n        data-hide-arrow={hideArrow}\n        onClick={(e) => {\n          e.stopPropagation();\n          e.preventDefault();\n          ref.current?.focus();\n          if (isOpen) {\n            onCloseFolder(id);\n          } else {\n            onOpenFolder(id);\n          }\n        }}\n      >\n        <ArrowDownShortLine />\n      </OpenButton>\n      <StyledName>{name}</StyledName>\n    </FolderNameLink>\n  ) : (\n    <FolderName\n      tabIndex={-1}\n      role=\"treeitem\"\n      id={treestructureId(type, folder.id)}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-selected={selected}\n      data-focused={focusedFolder?.id === folder.id}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      aria-label={`${name}${folder.status === \"shared\" ? `, ${t(\"myNdla.folder.sharing.shared\")}` : \"\"}`}\n      variant=\"ghost\"\n      shape=\"sharp\"\n      fontWeight=\"normal\"\n      colorTheme=\"light\"\n      ref={ref}\n      style={levelVariable}\n      data-selected={selected}\n      disabled={loading}\n      onFocus={() => setFocusedFolder(focusedFolder || folder)}\n      onClick={handleClickFolder}\n      data-creating={isCreatingFolder}\n    >\n      <IconWrapper>\n        <OpenButton\n          aria-hidden\n          tabIndex={-1}\n          data-open={isOpen}\n          data-hide-arrow={hideArrow}\n          onClick={(e) => {\n            e.stopPropagation();\n            setFocusedFolder(folder);\n            if (isOpen) {\n              onCloseFolder(id);\n            } else {\n              onOpenFolder(id);\n            }\n          }}\n        >\n          <ArrowDownShortLine />\n        </OpenButton>\n        <FolderIconWrapper>\n          <FolderIcon />\n        </FolderIconWrapper>\n      </IconWrapper>\n      <StyledName>{name}</StyledName>\n      {containsResource && (\n        <StyledCheck\n          aria-label={t(\"myNdla.alreadyInFolder\")}\n          id={`alreadyAdded-${folder.id}`}\n          title={t(\"myNdla.alreadyInFolder\")}\n        />\n      )}\n    </FolderName>\n  );\n};\n\nexport default FolderItem;\n"]} */"));
70
- const FolderNameLink = /*#__PURE__*/_styled(SafeLink, {
71
- target: "e11ok6h80",
72
- label: "FolderNameLink"
73
- })("display:grid;align-items:center;grid-template-columns:", spacing.medium, " 1fr auto;padding:", spacing.small, " ", spacing.xxsmall, ";padding-left:calc(0.75 * ", spacing.normal, " * var(--level));gap:", spacing.xxsmall, ";cursor:pointer;border:none;box-shadow:none;color:", colors.text.primary, ";", fonts.sizes("16px"), ";transition:", animations.durations.superFast, ";word-break:break-word;&[data-selected=\"true\"]{color:", colors.brand.primary, ";font-weight:", fonts.weight.semibold, ";}&:hover,&:focus{color:", colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FolderItem.tsx"],"names":[],"mappings":"AA6GuC","file":"FolderItem.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 { CSSProperties, KeyboardEvent, useEffect, useMemo, useRef } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { ButtonV2 as Button } from \"@ndla/button\";\nimport { colors, spacing, animations, misc, fonts } from \"@ndla/core\";\nimport { ArrowDownShortLine } from \"@ndla/icons/common\";\nimport { FolderUserFill } from \"@ndla/icons/contentType\";\nimport { CheckLine, FolderLine } from \"@ndla/icons/editor\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport { IFolder } from \"@ndla/types-backend/myndla-api\";\nimport { arrowNavigation } from \"./arrowNavigation\";\nimport { treestructureId } from \"./helperFunctions\";\nimport { CommonFolderItemsProps } from \"./types\";\n\nconst OpenButton = styled.span`\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  align-self: stretch;\n  color: ${colors.brand.tertiary};\n  ${misc.transition.default};\n  cursor: pointer;\n  &:hover {\n    color: ${colors.brand.primary};\n  }\n  svg {\n    width: 24px;\n    height: 24px;\n    transform: rotate(-90deg);\n  }\n  &[data-open=\"true\"] {\n    svg {\n      transform: rotate(0deg);\n    }\n  }\n  &[data-hide-arrow=\"true\"] {\n    visibility: hidden;\n  }\n`;\n\nconst StyledName = styled.span`\n  white-space: nowrap;\n  overflow: hidden;\n  text-overflow: ellipsis;\n  grid-column-start: 2;\n  text-align: left;\n`;\n\nconst IconWrapper = styled.div`\n  display: flex;\n`;\n\nconst FolderIconWrapper = styled.div`\n  svg {\n    height: 24px;\n    width: 24px;\n  }\n`;\n\nconst FolderName = styled(Button)`\n  display: grid;\n  grid-template-columns: auto 1fr auto;\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  border: none;\n  outline: none;\n  color: ${colors.text.primary};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n\n  &:hover {\n    box-shadow: none;\n    outline: none;\n    background: ${colors.brand.lightest};\n    color: ${colors.text.primary};\n  }\n\n  &[data-focused=\"true\"] {\n    background: ${colors.brand.lightest};\n  }\n\n  &[data-selected=\"true\"] {\n    background: ${colors.brand.lighter};\n    &:hover {\n      background: ${colors.brand.light};\n    }\n  }\n\n  &[data-creating=\"true\"][data-focused=\"true\"] {\n    color: ${colors.brand.primary};\n  }\n\n  &[data-creating=\"true\"] {\n    background: none;\n  }\n`;\n\nconst StyledCheck = styled(CheckLine)`\n  color: ${colors.support.green};\n`;\n\nconst FolderNameLink = styled(SafeLink)`\n  display: grid;\n  align-items: center;\n  grid-template-columns: ${spacing.medium} 1fr auto;\n  padding: ${spacing.small} ${spacing.xxsmall};\n  padding-left: calc(0.75 * ${spacing.normal} * var(--level));\n  gap: ${spacing.xxsmall};\n  cursor: pointer;\n\n  border: none;\n  box-shadow: none;\n  color: ${colors.text.primary};\n  ${fonts.sizes(\"16px\")};\n  transition: ${animations.durations.superFast};\n  word-break: break-word;\n  &[data-selected=\"true\"] {\n    color: ${colors.brand.primary};\n    font-weight: ${fonts.weight.semibold};\n  }\n  &:hover,\n  &:focus {\n    color: ${colors.brand.primary};\n  }\n`;\n\ninterface Props extends CommonFolderItemsProps {\n  isOpen: boolean;\n  folder: IFolder;\n  isCreatingFolder?: boolean;\n  index: number;\n}\n\nconst FolderItem = ({\n  focusedFolder,\n  folder,\n  isOpen,\n  level,\n  loading,\n  selectedFolder,\n  onCloseFolder,\n  onOpenFolder,\n  setFocusedFolder,\n  setSelectedFolder,\n  targetResource,\n  visibleFolders,\n  maxLevel,\n  isCreatingFolder,\n  type,\n  closeTree,\n  index,\n}: Props) => {\n  const { t } = useTranslation();\n  const { id, name } = folder;\n  const ref = useRef<HTMLButtonElement & HTMLAnchorElement>(null);\n  const selected = selectedFolder ? selectedFolder.id === id : false;\n\n  const levelVariable = useMemo(() => ({ \"--level\": level }) as unknown as CSSProperties, [level]);\n\n  const focused = focusedFolder?.id === id;\n\n  const handleClickFolder = () => {\n    if (!selected) {\n      setSelectedFolder(folder);\n    }\n    setFocusedFolder(folder);\n    if (type === \"picker\") {\n      if (selected) {\n        closeTree();\n      }\n    }\n  };\n\n  useEffect(() => {\n    if (focusedFolder?.id === id && !isCreatingFolder) {\n      if (type === \"navigation\") {\n        ref.current?.focus();\n      }\n      if (type === \"picker\") {\n        ref.current?.scrollIntoView({\n          behavior: \"smooth\",\n          block: \"start\",\n        });\n      }\n    }\n  }, [focusedFolder, ref, id, isCreatingFolder, type]);\n\n  const linkPath = `/minndla/folders/${id}`;\n\n  const containsResource =\n    targetResource && folder.resources.some((resource) => resource.resourceId === targetResource.resourceId);\n\n  const emptyFolder = folder.subfolders.length === 0;\n  const isMaxDepth = level > maxLevel;\n  const hideArrow = isMaxDepth || emptyFolder;\n\n  const FolderIcon = folder.status === \"shared\" ? FolderUserFill : FolderLine;\n\n  const tabable = selected || focused || (!focusedFolder && !folder.parentId && index === 0);\n\n  return type === \"navigation\" ? (\n    <FolderNameLink\n      role=\"treeitem\"\n      aria-owns={folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-current={selected ? \"page\" : undefined}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      ref={ref}\n      style={levelVariable}\n      onKeyDown={(e: KeyboardEvent<HTMLElement>) => {\n        if (e.key === \"Enter\") {\n          setSelectedFolder(folder);\n          return;\n        }\n        arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);\n      }}\n      to={loading ? \"\" : linkPath}\n      tabIndex={tabable ? 0 : -1}\n      data-selected={selected}\n      onFocus={() => setFocusedFolder(folder)}\n      onClick={handleClickFolder}\n    >\n      <OpenButton\n        aria-hidden\n        tabIndex={-1}\n        data-open={isOpen}\n        data-hide-arrow={hideArrow}\n        onClick={(e) => {\n          e.stopPropagation();\n          e.preventDefault();\n          ref.current?.focus();\n          if (isOpen) {\n            onCloseFolder(id);\n          } else {\n            onOpenFolder(id);\n          }\n        }}\n      >\n        <ArrowDownShortLine />\n      </OpenButton>\n      <StyledName>{name}</StyledName>\n    </FolderNameLink>\n  ) : (\n    <FolderName\n      tabIndex={-1}\n      role=\"treeitem\"\n      id={treestructureId(type, folder.id)}\n      aria-expanded={isMaxDepth || emptyFolder ? undefined : isOpen}\n      aria-selected={selected}\n      data-focused={focusedFolder?.id === folder.id}\n      aria-describedby={containsResource ? `alreadyAdded-${folder.id}` : undefined}\n      aria-label={`${name}${folder.status === \"shared\" ? `, ${t(\"myNdla.folder.sharing.shared\")}` : \"\"}`}\n      variant=\"ghost\"\n      shape=\"sharp\"\n      fontWeight=\"normal\"\n      colorTheme=\"light\"\n      ref={ref}\n      style={levelVariable}\n      data-selected={selected}\n      disabled={loading}\n      onFocus={() => setFocusedFolder(focusedFolder || folder)}\n      onClick={handleClickFolder}\n      data-creating={isCreatingFolder}\n    >\n      <IconWrapper>\n        <OpenButton\n          aria-hidden\n          tabIndex={-1}\n          data-open={isOpen}\n          data-hide-arrow={hideArrow}\n          onClick={(e) => {\n            e.stopPropagation();\n            setFocusedFolder(folder);\n            if (isOpen) {\n              onCloseFolder(id);\n            } else {\n              onOpenFolder(id);\n            }\n          }}\n        >\n          <ArrowDownShortLine />\n        </OpenButton>\n        <FolderIconWrapper>\n          <FolderIcon />\n        </FolderIconWrapper>\n      </IconWrapper>\n      <StyledName>{name}</StyledName>\n      {containsResource && (\n        <StyledCheck\n          aria-label={t(\"myNdla.alreadyInFolder\")}\n          id={`alreadyAdded-${folder.id}`}\n          title={t(\"myNdla.alreadyInFolder\")}\n        />\n      )}\n    </FolderName>\n  );\n};\n\nexport default FolderItem;\n"]} */"));
74
- const FolderItem = _ref => {
75
- let {
76
- focusedFolder,
77
- folder,
78
- isOpen,
79
- level,
80
- loading,
81
- selectedFolder,
82
- onCloseFolder,
83
- onOpenFolder,
84
- setFocusedFolder,
85
- setSelectedFolder,
86
- targetResource,
87
- visibleFolders,
88
- maxLevel,
89
- isCreatingFolder,
90
- type,
91
- closeTree,
92
- index
93
- } = _ref;
94
- const {
95
- t
96
- } = useTranslation();
97
- const {
98
- id,
99
- name
100
- } = folder;
101
- const ref = useRef(null);
102
- const selected = selectedFolder ? selectedFolder.id === id : false;
103
- const levelVariable = useMemo(() => ({
104
- "--level": level
105
- }), [level]);
106
- const focused = focusedFolder?.id === id;
107
- const handleClickFolder = () => {
108
- if (!selected) {
109
- setSelectedFolder(folder);
110
- }
111
- setFocusedFolder(folder);
112
- if (type === "picker") {
113
- if (selected) {
114
- closeTree();
115
- }
116
- }
117
- };
118
- useEffect(() => {
119
- if (focusedFolder?.id === id && !isCreatingFolder) {
120
- if (type === "navigation") {
121
- ref.current?.focus();
122
- }
123
- if (type === "picker") {
124
- ref.current?.scrollIntoView({
125
- behavior: "smooth",
126
- block: "start"
127
- });
128
- }
129
- }
130
- }, [focusedFolder, ref, id, isCreatingFolder, type]);
131
- const linkPath = `/minndla/folders/${id}`;
132
- const containsResource = targetResource && folder.resources.some(resource => resource.resourceId === targetResource.resourceId);
133
- const emptyFolder = folder.subfolders.length === 0;
134
- const isMaxDepth = level > maxLevel;
135
- const hideArrow = isMaxDepth || emptyFolder;
136
- const FolderIcon = folder.status === "shared" ? FolderUserFill : FolderLine;
137
- const tabable = selected || focused || !focusedFolder && !folder.parentId && index === 0;
138
- return type === "navigation" ? /*#__PURE__*/_jsxs(FolderNameLink, {
139
- role: "treeitem",
140
- "aria-owns": folder.subfolders.length ? treestructureId(type, `subfolders-${folder.id}`) : undefined,
141
- "aria-expanded": isMaxDepth || emptyFolder ? undefined : isOpen,
142
- "aria-current": selected ? "page" : undefined,
143
- "aria-describedby": containsResource ? `alreadyAdded-${folder.id}` : undefined,
144
- ref: ref,
145
- style: levelVariable,
146
- onKeyDown: e => {
147
- if (e.key === "Enter") {
148
- setSelectedFolder(folder);
149
- return;
150
- }
151
- arrowNavigation(e, id, visibleFolders, setFocusedFolder, onOpenFolder, onCloseFolder);
152
- },
153
- to: loading ? "" : linkPath,
154
- tabIndex: tabable ? 0 : -1,
155
- "data-selected": selected,
156
- onFocus: () => setFocusedFolder(folder),
157
- onClick: handleClickFolder,
158
- children: [/*#__PURE__*/_jsx(OpenButton, {
159
- "aria-hidden": true,
160
- tabIndex: -1,
161
- "data-open": isOpen,
162
- "data-hide-arrow": hideArrow,
163
- onClick: e => {
164
- e.stopPropagation();
165
- e.preventDefault();
166
- ref.current?.focus();
167
- if (isOpen) {
168
- onCloseFolder(id);
169
- } else {
170
- onOpenFolder(id);
171
- }
172
- },
173
- children: /*#__PURE__*/_jsx(ArrowDownShortLine, {})
174
- }), /*#__PURE__*/_jsx(StyledName, {
175
- children: name
176
- })]
177
- }) : /*#__PURE__*/_jsxs(FolderName, {
178
- tabIndex: -1,
179
- role: "treeitem",
180
- id: treestructureId(type, folder.id),
181
- "aria-expanded": isMaxDepth || emptyFolder ? undefined : isOpen,
182
- "aria-selected": selected,
183
- "data-focused": focusedFolder?.id === folder.id,
184
- "aria-describedby": containsResource ? `alreadyAdded-${folder.id}` : undefined,
185
- "aria-label": `${name}${folder.status === "shared" ? `, ${t("myNdla.folder.sharing.shared")}` : ""}`,
186
- variant: "ghost",
187
- shape: "sharp",
188
- fontWeight: "normal",
189
- colorTheme: "light",
190
- ref: ref,
191
- style: levelVariable,
192
- "data-selected": selected,
193
- disabled: loading,
194
- onFocus: () => setFocusedFolder(focusedFolder || folder),
195
- onClick: handleClickFolder,
196
- "data-creating": isCreatingFolder,
197
- children: [/*#__PURE__*/_jsxs(IconWrapper, {
198
- children: [/*#__PURE__*/_jsx(OpenButton, {
199
- "aria-hidden": true,
200
- tabIndex: -1,
201
- "data-open": isOpen,
202
- "data-hide-arrow": hideArrow,
203
- onClick: e => {
204
- e.stopPropagation();
205
- setFocusedFolder(folder);
206
- if (isOpen) {
207
- onCloseFolder(id);
208
- } else {
209
- onOpenFolder(id);
210
- }
211
- },
212
- children: /*#__PURE__*/_jsx(ArrowDownShortLine, {})
213
- }), /*#__PURE__*/_jsx(FolderIconWrapper, {
214
- children: /*#__PURE__*/_jsx(FolderIcon, {})
215
- })]
216
- }), /*#__PURE__*/_jsx(StyledName, {
217
- children: name
218
- }), containsResource && /*#__PURE__*/_jsx(StyledCheck, {
219
- "aria-label": t("myNdla.alreadyInFolder"),
220
- id: `alreadyAdded-${folder.id}`,
221
- title: t("myNdla.alreadyInFolder")
222
- })]
223
- });
224
- };
225
- export default FolderItem;
@@ -1,95 +0,0 @@
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
- /**
4
- * Copyright (c) 2022-present, NDLA.
5
- *
6
- * This source code is licensed under the GPLv3 license found in the
7
- * LICENSE file in the root directory of this source tree.
8
- *
9
- */
10
-
11
- import { animations } from "@ndla/core";
12
- import FolderItem from "./FolderItem";
13
- import { treestructureId } from "./helperFunctions";
14
- import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
15
- const StyledUL = /*#__PURE__*/_styled("ul", {
16
- target: "ef48q151",
17
- label: "StyledUL"
18
- })(animations.fadeInLeft(animations.durations.fast), ";animation-fill-mode:forwards;@media (prefers-reduced-motion: reduce){animation:none;}list-style:none;padding:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkZvbGRlckl0ZW1zLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFnQjBCIiwiZmlsZSI6IkZvbGRlckl0ZW1zLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDIyLXByZXNlbnQsIE5ETEEuXG4gKlxuICogVGhpcyBzb3VyY2UgY29kZSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgR1BMdjMgbGljZW5zZSBmb3VuZCBpbiB0aGVcbiAqIExJQ0VOU0UgZmlsZSBpbiB0aGUgcm9vdCBkaXJlY3Rvcnkgb2YgdGhpcyBzb3VyY2UgdHJlZS5cbiAqXG4gKi9cblxuaW1wb3J0IHsgUmVhY3ROb2RlIH0gZnJvbSBcInJlYWN0XCI7XG5pbXBvcnQgc3R5bGVkIGZyb20gXCJAZW1vdGlvbi9zdHlsZWRcIjtcbmltcG9ydCB7IGFuaW1hdGlvbnMgfSBmcm9tIFwiQG5kbGEvY29yZVwiO1xuaW1wb3J0IHsgSUZvbGRlciB9IGZyb20gXCJAbmRsYS90eXBlcy1iYWNrZW5kL215bmRsYS1hcGlcIjtcbmltcG9ydCBGb2xkZXJJdGVtIGZyb20gXCIuL0ZvbGRlckl0ZW1cIjtcbmltcG9ydCB7IHRyZWVzdHJ1Y3R1cmVJZCB9IGZyb20gXCIuL2hlbHBlckZ1bmN0aW9uc1wiO1xuaW1wb3J0IHsgQ29tbW9uRm9sZGVySXRlbXNQcm9wcywgTmV3Rm9sZGVySW5wdXRGdW5jLCBPbkNyZWF0ZWRGdW5jIH0gZnJvbSBcIi4vdHlwZXNcIjtcblxuY29uc3QgU3R5bGVkVUwgPSBzdHlsZWQudWxgXG4gICR7YW5pbWF0aW9ucy5mYWRlSW5MZWZ0KGFuaW1hdGlvbnMuZHVyYXRpb25zLmZhc3QpfTtcbiAgYW5pbWF0aW9uLWZpbGwtbW9kZTogZm9yd2FyZHM7XG4gIEBtZWRpYSAocHJlZmVycy1yZWR1Y2VkLW1vdGlvbjogcmVkdWNlKSB7XG4gICAgYW5pbWF0aW9uOiBub25lO1xuICB9XG4gIGxpc3Qtc3R5bGU6IG5vbmU7XG4gIHBhZGRpbmc6IDA7XG5gO1xuXG5jb25zdCBTdHlsZWRMSSA9IHN0eWxlZC5saWBcbiAgZGlzcGxheTogZmxleDtcbiAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgcGFkZGluZzogMDtcbiAgJltkYXRhLXR5cGU9XCJuYXZpZ2F0aW9uXCJdIHtcbiAgICBhbGlnbi1pdGVtczogZmxleC1zdGFydDtcbiAgfVxuYDtcblxuZXhwb3J0IGludGVyZmFjZSBGb2xkZXJJdGVtc1Byb3BzIGV4dGVuZHMgQ29tbW9uRm9sZGVySXRlbXNQcm9wcyB7XG4gIGZvbGRlcnM6IElGb2xkZXJbXTtcbiAgbmV3Rm9sZGVyUGFyZW50SWQ6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgb25DYW5jZWxOZXdGb2xkZXI6ICgpID0+IHZvaWQ7XG4gIG9wZW5Gb2xkZXJzOiBzdHJpbmdbXTtcbiAgcGFyZW50Rm9sZGVyPzogSUZvbGRlcjtcbiAgY2hpbGRyZW4/OiBSZWFjdE5vZGU7XG4gIG9uQ3JlYXRlOiBPbkNyZWF0ZWRGdW5jO1xuICBuZXdGb2xkZXJJbnB1dD86IE5ld0ZvbGRlcklucHV0RnVuYztcbn1cblxuY29uc3QgRm9sZGVySXRlbXMgPSAoe1xuICBmb2xkZXJzLFxuICBsZXZlbCxcbiAgbG9hZGluZyxcbiAgbmV3Rm9sZGVyUGFyZW50SWQsXG4gIG9uQ2FuY2VsTmV3Rm9sZGVyLFxuICBvcGVuRm9sZGVycyxcbiAgdHlwZSxcbiAgcGFyZW50Rm9sZGVyLFxuICBjaGlsZHJlbixcbiAgb25DcmVhdGUsXG4gIG5ld0ZvbGRlcklucHV0LFxuICAuLi5yZXN0XG59OiBGb2xkZXJJdGVtc1Byb3BzKSA9PiAoXG4gIDxTdHlsZWRVTFxuICAgIGlkPXtcbiAgICAgIGxldmVsID09PSAwICYmIHR5cGUgPT09IFwicGlja2VyXCJcbiAgICAgICAgPyB0cmVlc3RydWN0dXJlSWQodHlwZSwgXCJwb3B1cFwiKVxuICAgICAgICA6IHBhcmVudEZvbGRlclxuICAgICAgICAgID8gdHJlZXN0cnVjdHVyZUlkKHR5cGUsIGBzdWJmb2xkZXJzLSR7cGFyZW50Rm9sZGVyLmlkfWApXG4gICAgICAgICAgOiB1bmRlZmluZWRcbiAgICB9XG4gICAgdGFiSW5kZXg9ey0xfVxuICAgIGFyaWEtbGFiZWxsZWRieT17bGV2ZWwgPT09IDAgJiYgdHlwZSA9PT0gXCJwaWNrZXJcIiA/IHRyZWVzdHJ1Y3R1cmVJZCh0eXBlLCBcImxhYmVsXCIpIDogdW5kZWZpbmVkfVxuICAgIHJvbGU9e2xldmVsID09PSAwID8gXCJ0cmVlXCIgOiBcImdyb3VwXCJ9XG4gID5cbiAgICB7Y2hpbGRyZW59XG4gICAge2ZvbGRlcnMubWFwKChmb2xkZXIsIGluZGV4KSA9PiB7XG4gICAgICBjb25zdCB7IHN1YmZvbGRlcnMsIGlkIH0gPSBmb2xkZXI7XG4gICAgICBjb25zdCBpc09wZW4gPSBvcGVuRm9sZGVycz8uaW5jbHVkZXMoaWQpO1xuXG4gICAgICByZXR1cm4gKFxuICAgICAgICA8U3R5bGVkTEkga2V5PXtpZH0gdGFiSW5kZXg9ey0xfSByb2xlPVwibm9uZVwiIGRhdGEtdHlwZT17dHlwZX0+XG4gICAgICAgICAgPEZvbGRlckl0ZW1cbiAgICAgICAgICAgIGluZGV4PXtpbmRleH1cbiAgICAgICAgICAgIGZvbGRlcj17Zm9sZGVyfVxuICAgICAgICAgICAgaXNPcGVuPXtpc09wZW59XG4gICAgICAgICAgICBsZXZlbD17bGV2ZWx9XG4gICAgICAgICAgICBsb2FkaW5nPXtsb2FkaW5nfVxuICAgICAgICAgICAgdHlwZT17dHlwZX1cbiAgICAgICAgICAgIGlzQ3JlYXRpbmdGb2xkZXI9eyEhbmV3Rm9sZGVyUGFyZW50SWR9XG4gICAgICAgICAgICB7Li4ucmVzdH1cbiAgICAgICAgICAvPlxuICAgICAgICAgIHsoKHN1YmZvbGRlcnMgJiYgaXNPcGVuKSB8fCBuZXdGb2xkZXJQYXJlbnRJZCA9PT0gaWQpICYmIChcbiAgICAgICAgICAgIDxGb2xkZXJJdGVtc1xuICAgICAgICAgICAgICBwYXJlbnRGb2xkZXI9e2ZvbGRlcn1cbiAgICAgICAgICAgICAgZm9sZGVycz17c3ViZm9sZGVyc31cbiAgICAgICAgICAgICAgbGV2ZWw9e2xldmVsICsgMX1cbiAgICAgICAgICAgICAgbG9hZGluZz17bG9hZGluZ31cbiAgICAgICAgICAgICAgdHlwZT17dHlwZX1cbiAgICAgICAgICAgICAgbmV3Rm9sZGVyUGFyZW50SWQ9e25ld0ZvbGRlclBhcmVudElkfVxuICAgICAgICAgICAgICBvbkNhbmNlbE5ld0ZvbGRlcj17b25DYW5jZWxOZXdGb2xkZXJ9XG4gICAgICAgICAgICAgIG9wZW5Gb2xkZXJzPXtvcGVuRm9sZGVyc31cbiAgICAgICAgICAgICAgbmV3Rm9sZGVySW5wdXQ9e25ld0ZvbGRlcklucHV0fVxuICAgICAgICAgICAgICBvbkNyZWF0ZT17b25DcmVhdGV9XG4gICAgICAgICAgICAgIHsuLi5yZXN0fVxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICB7bmV3Rm9sZGVyUGFyZW50SWQgPT09IGlkICYmIChcbiAgICAgICAgICAgICAgICA8bGkgcm9sZT1cIm5vbmVcIj5cbiAgICAgICAgICAgICAgICAgIHtuZXdGb2xkZXJJbnB1dD8uKHtcbiAgICAgICAgICAgICAgICAgICAgcGFyZW50SWQ6IGlkLFxuICAgICAgICAgICAgICAgICAgICBvbkNsb3NlOiBvbkNhbmNlbE5ld0ZvbGRlcixcbiAgICAgICAgICAgICAgICAgICAgb25DcmVhdGUsXG4gICAgICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgICAgICA8L2xpPlxuICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgPC9Gb2xkZXJJdGVtcz5cbiAgICAgICAgICApfVxuICAgICAgICA8L1N0eWxlZExJPlxuICAgICAgKTtcbiAgICB9KX1cbiAgPC9TdHlsZWRVTD5cbik7XG5cbmV4cG9ydCBkZWZhdWx0IEZvbGRlckl0ZW1zO1xuIl19 */"));
19
- const StyledLI = /*#__PURE__*/_styled("li", {
20
- target: "ef48q150",
21
- label: "StyledLI"
22
- })(process.env.NODE_ENV === "production" ? {
23
- name: "snw2uv",
24
- styles: "display:flex;flex-direction:column;padding:0;&[data-type=\"navigation\"]{align-items:flex-start;}"
25
- } : {
26
- name: "snw2uv",
27
- styles: "display:flex;flex-direction:column;padding:0;&[data-type=\"navigation\"]{align-items:flex-start;}",
28
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIkZvbGRlckl0ZW1zLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUEwQjBCIiwiZmlsZSI6IkZvbGRlckl0ZW1zLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDIyLXByZXNlbnQsIE5ETEEuXG4gKlxuICogVGhpcyBzb3VyY2UgY29kZSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgR1BMdjMgbGljZW5zZSBmb3VuZCBpbiB0aGVcbiAqIExJQ0VOU0UgZmlsZSBpbiB0aGUgcm9vdCBkaXJlY3Rvcnkgb2YgdGhpcyBzb3VyY2UgdHJlZS5cbiAqXG4gKi9cblxuaW1wb3J0IHsgUmVhY3ROb2RlIH0gZnJvbSBcInJlYWN0XCI7XG5pbXBvcnQgc3R5bGVkIGZyb20gXCJAZW1vdGlvbi9zdHlsZWRcIjtcbmltcG9ydCB7IGFuaW1hdGlvbnMgfSBmcm9tIFwiQG5kbGEvY29yZVwiO1xuaW1wb3J0IHsgSUZvbGRlciB9IGZyb20gXCJAbmRsYS90eXBlcy1iYWNrZW5kL215bmRsYS1hcGlcIjtcbmltcG9ydCBGb2xkZXJJdGVtIGZyb20gXCIuL0ZvbGRlckl0ZW1cIjtcbmltcG9ydCB7IHRyZWVzdHJ1Y3R1cmVJZCB9IGZyb20gXCIuL2hlbHBlckZ1bmN0aW9uc1wiO1xuaW1wb3J0IHsgQ29tbW9uRm9sZGVySXRlbXNQcm9wcywgTmV3Rm9sZGVySW5wdXRGdW5jLCBPbkNyZWF0ZWRGdW5jIH0gZnJvbSBcIi4vdHlwZXNcIjtcblxuY29uc3QgU3R5bGVkVUwgPSBzdHlsZWQudWxgXG4gICR7YW5pbWF0aW9ucy5mYWRlSW5MZWZ0KGFuaW1hdGlvbnMuZHVyYXRpb25zLmZhc3QpfTtcbiAgYW5pbWF0aW9uLWZpbGwtbW9kZTogZm9yd2FyZHM7XG4gIEBtZWRpYSAocHJlZmVycy1yZWR1Y2VkLW1vdGlvbjogcmVkdWNlKSB7XG4gICAgYW5pbWF0aW9uOiBub25lO1xuICB9XG4gIGxpc3Qtc3R5bGU6IG5vbmU7XG4gIHBhZGRpbmc6IDA7XG5gO1xuXG5jb25zdCBTdHlsZWRMSSA9IHN0eWxlZC5saWBcbiAgZGlzcGxheTogZmxleDtcbiAgZmxleC1kaXJlY3Rpb246IGNvbHVtbjtcbiAgcGFkZGluZzogMDtcbiAgJltkYXRhLXR5cGU9XCJuYXZpZ2F0aW9uXCJdIHtcbiAgICBhbGlnbi1pdGVtczogZmxleC1zdGFydDtcbiAgfVxuYDtcblxuZXhwb3J0IGludGVyZmFjZSBGb2xkZXJJdGVtc1Byb3BzIGV4dGVuZHMgQ29tbW9uRm9sZGVySXRlbXNQcm9wcyB7XG4gIGZvbGRlcnM6IElGb2xkZXJbXTtcbiAgbmV3Rm9sZGVyUGFyZW50SWQ6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgb25DYW5jZWxOZXdGb2xkZXI6ICgpID0+IHZvaWQ7XG4gIG9wZW5Gb2xkZXJzOiBzdHJpbmdbXTtcbiAgcGFyZW50Rm9sZGVyPzogSUZvbGRlcjtcbiAgY2hpbGRyZW4/OiBSZWFjdE5vZGU7XG4gIG9uQ3JlYXRlOiBPbkNyZWF0ZWRGdW5jO1xuICBuZXdGb2xkZXJJbnB1dD86IE5ld0ZvbGRlcklucHV0RnVuYztcbn1cblxuY29uc3QgRm9sZGVySXRlbXMgPSAoe1xuICBmb2xkZXJzLFxuICBsZXZlbCxcbiAgbG9hZGluZyxcbiAgbmV3Rm9sZGVyUGFyZW50SWQsXG4gIG9uQ2FuY2VsTmV3Rm9sZGVyLFxuICBvcGVuRm9sZGVycyxcbiAgdHlwZSxcbiAgcGFyZW50Rm9sZGVyLFxuICBjaGlsZHJlbixcbiAgb25DcmVhdGUsXG4gIG5ld0ZvbGRlcklucHV0LFxuICAuLi5yZXN0XG59OiBGb2xkZXJJdGVtc1Byb3BzKSA9PiAoXG4gIDxTdHlsZWRVTFxuICAgIGlkPXtcbiAgICAgIGxldmVsID09PSAwICYmIHR5cGUgPT09IFwicGlja2VyXCJcbiAgICAgICAgPyB0cmVlc3RydWN0dXJlSWQodHlwZSwgXCJwb3B1cFwiKVxuICAgICAgICA6IHBhcmVudEZvbGRlclxuICAgICAgICAgID8gdHJlZXN0cnVjdHVyZUlkKHR5cGUsIGBzdWJmb2xkZXJzLSR7cGFyZW50Rm9sZGVyLmlkfWApXG4gICAgICAgICAgOiB1bmRlZmluZWRcbiAgICB9XG4gICAgdGFiSW5kZXg9ey0xfVxuICAgIGFyaWEtbGFiZWxsZWRieT17bGV2ZWwgPT09IDAgJiYgdHlwZSA9PT0gXCJwaWNrZXJcIiA/IHRyZWVzdHJ1Y3R1cmVJZCh0eXBlLCBcImxhYmVsXCIpIDogdW5kZWZpbmVkfVxuICAgIHJvbGU9e2xldmVsID09PSAwID8gXCJ0cmVlXCIgOiBcImdyb3VwXCJ9XG4gID5cbiAgICB7Y2hpbGRyZW59XG4gICAge2ZvbGRlcnMubWFwKChmb2xkZXIsIGluZGV4KSA9PiB7XG4gICAgICBjb25zdCB7IHN1YmZvbGRlcnMsIGlkIH0gPSBmb2xkZXI7XG4gICAgICBjb25zdCBpc09wZW4gPSBvcGVuRm9sZGVycz8uaW5jbHVkZXMoaWQpO1xuXG4gICAgICByZXR1cm4gKFxuICAgICAgICA8U3R5bGVkTEkga2V5PXtpZH0gdGFiSW5kZXg9ey0xfSByb2xlPVwibm9uZVwiIGRhdGEtdHlwZT17dHlwZX0+XG4gICAgICAgICAgPEZvbGRlckl0ZW1cbiAgICAgICAgICAgIGluZGV4PXtpbmRleH1cbiAgICAgICAgICAgIGZvbGRlcj17Zm9sZGVyfVxuICAgICAgICAgICAgaXNPcGVuPXtpc09wZW59XG4gICAgICAgICAgICBsZXZlbD17bGV2ZWx9XG4gICAgICAgICAgICBsb2FkaW5nPXtsb2FkaW5nfVxuICAgICAgICAgICAgdHlwZT17dHlwZX1cbiAgICAgICAgICAgIGlzQ3JlYXRpbmdGb2xkZXI9eyEhbmV3Rm9sZGVyUGFyZW50SWR9XG4gICAgICAgICAgICB7Li4ucmVzdH1cbiAgICAgICAgICAvPlxuICAgICAgICAgIHsoKHN1YmZvbGRlcnMgJiYgaXNPcGVuKSB8fCBuZXdGb2xkZXJQYXJlbnRJZCA9PT0gaWQpICYmIChcbiAgICAgICAgICAgIDxGb2xkZXJJdGVtc1xuICAgICAgICAgICAgICBwYXJlbnRGb2xkZXI9e2ZvbGRlcn1cbiAgICAgICAgICAgICAgZm9sZGVycz17c3ViZm9sZGVyc31cbiAgICAgICAgICAgICAgbGV2ZWw9e2xldmVsICsgMX1cbiAgICAgICAgICAgICAgbG9hZGluZz17bG9hZGluZ31cbiAgICAgICAgICAgICAgdHlwZT17dHlwZX1cbiAgICAgICAgICAgICAgbmV3Rm9sZGVyUGFyZW50SWQ9e25ld0ZvbGRlclBhcmVudElkfVxuICAgICAgICAgICAgICBvbkNhbmNlbE5ld0ZvbGRlcj17b25DYW5jZWxOZXdGb2xkZXJ9XG4gICAgICAgICAgICAgIG9wZW5Gb2xkZXJzPXtvcGVuRm9sZGVyc31cbiAgICAgICAgICAgICAgbmV3Rm9sZGVySW5wdXQ9e25ld0ZvbGRlcklucHV0fVxuICAgICAgICAgICAgICBvbkNyZWF0ZT17b25DcmVhdGV9XG4gICAgICAgICAgICAgIHsuLi5yZXN0fVxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICB7bmV3Rm9sZGVyUGFyZW50SWQgPT09IGlkICYmIChcbiAgICAgICAgICAgICAgICA8bGkgcm9sZT1cIm5vbmVcIj5cbiAgICAgICAgICAgICAgICAgIHtuZXdGb2xkZXJJbnB1dD8uKHtcbiAgICAgICAgICAgICAgICAgICAgcGFyZW50SWQ6IGlkLFxuICAgICAgICAgICAgICAgICAgICBvbkNsb3NlOiBvbkNhbmNlbE5ld0ZvbGRlcixcbiAgICAgICAgICAgICAgICAgICAgb25DcmVhdGUsXG4gICAgICAgICAgICAgICAgICB9KX1cbiAgICAgICAgICAgICAgICA8L2xpPlxuICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgPC9Gb2xkZXJJdGVtcz5cbiAgICAgICAgICApfVxuICAgICAgICA8L1N0eWxlZExJPlxuICAgICAgKTtcbiAgICB9KX1cbiAgPC9TdHlsZWRVTD5cbik7XG5cbmV4cG9ydCBkZWZhdWx0IEZvbGRlckl0ZW1zO1xuIl19 */",
29
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
30
- });
31
- const FolderItems = _ref => {
32
- let {
33
- folders,
34
- level,
35
- loading,
36
- newFolderParentId,
37
- onCancelNewFolder,
38
- openFolders,
39
- type,
40
- parentFolder,
41
- children,
42
- onCreate,
43
- newFolderInput,
44
- ...rest
45
- } = _ref;
46
- return /*#__PURE__*/_jsxs(StyledUL, {
47
- id: level === 0 && type === "picker" ? treestructureId(type, "popup") : parentFolder ? treestructureId(type, `subfolders-${parentFolder.id}`) : undefined,
48
- tabIndex: -1,
49
- "aria-labelledby": level === 0 && type === "picker" ? treestructureId(type, "label") : undefined,
50
- role: level === 0 ? "tree" : "group",
51
- children: [children, folders.map((folder, index) => {
52
- const {
53
- subfolders,
54
- id
55
- } = folder;
56
- const isOpen = openFolders?.includes(id);
57
- return /*#__PURE__*/_jsxs(StyledLI, {
58
- tabIndex: -1,
59
- role: "none",
60
- "data-type": type,
61
- children: [/*#__PURE__*/_jsx(FolderItem, {
62
- index: index,
63
- folder: folder,
64
- isOpen: isOpen,
65
- level: level,
66
- loading: loading,
67
- type: type,
68
- isCreatingFolder: !!newFolderParentId,
69
- ...rest
70
- }), (subfolders && isOpen || newFolderParentId === id) && /*#__PURE__*/_jsx(FolderItems, {
71
- parentFolder: folder,
72
- folders: subfolders,
73
- level: level + 1,
74
- loading: loading,
75
- type: type,
76
- newFolderParentId: newFolderParentId,
77
- onCancelNewFolder: onCancelNewFolder,
78
- openFolders: openFolders,
79
- newFolderInput: newFolderInput,
80
- onCreate: onCreate,
81
- ...rest,
82
- children: newFolderParentId === id && /*#__PURE__*/_jsx("li", {
83
- role: "none",
84
- children: newFolderInput?.({
85
- parentId: id,
86
- onClose: onCancelNewFolder,
87
- onCreate
88
- })
89
- })
90
- })]
91
- }, id);
92
- })]
93
- });
94
- };
95
- export default FolderItems;
@@ -1,35 +0,0 @@
1
- /**
2
- * Copyright (c) 2022-present, NDLA.
3
- *
4
- * This source code is licensed under the GPLv3 license found in the
5
- * LICENSE file in the root directory of this source tree.
6
- *
7
- */
8
-
9
- const navigateVertical = (visibleFolders, folderId, setFocusedFolderId, direction) => {
10
- const currentIndex = visibleFolders.findIndex(folder => folder.id === folderId);
11
- const target = visibleFolders[currentIndex + direction];
12
- if (target !== undefined) {
13
- setFocusedFolderId(target);
14
- }
15
- };
16
- const arrowKeys = ["ArrowUp", "ArrowDown", "ArrowLeft", "ArrowRight", "Enter"];
17
- export const arrowNavigation = (e, id, visibleFolders, setFocusedFolderId, onOpen, onClose) => {
18
- if (!arrowKeys.includes(e.key)) {
19
- return;
20
- }
21
- e.preventDefault();
22
- e.stopPropagation();
23
- switch (e.key) {
24
- case "ArrowUp":
25
- return navigateVertical(visibleFolders, id, setFocusedFolderId, -1);
26
- case "ArrowDown":
27
- return navigateVertical(visibleFolders, id, setFocusedFolderId, 1);
28
- case "ArrowLeft":
29
- return onClose(id);
30
- case "ArrowRight":
31
- return onOpen(id);
32
- default:
33
- return;
34
- }
35
- };
@@ -1,23 +0,0 @@
1
- /**
2
- * Copyright (c) 2023-present, NDLA.
3
- *
4
- * This source code is licensed under the GPLv3 license found in the
5
- * LICENSE file in the root directory of this source tree.
6
- *
7
- */
8
- interface Image {
9
- src: string;
10
- alt: string;
11
- }
12
- export interface Programme {
13
- id: string;
14
- title: {
15
- title: string;
16
- language: string;
17
- };
18
- narrowImage?: Image;
19
- wideImage?: Image;
20
- url: string;
21
- }
22
- declare const ProgrammeCard: ({ title, narrowImage, wideImage, url }: Programme) => import("react/jsx-runtime").JSX.Element;
23
- export default ProgrammeCard;