@ndla/ui 56.0.13-alpha.0 → 56.0.15-alpha.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/panda.buildinfo.json +7 -2
- package/dist/styles.css +24 -4
- package/es/Article/Article.js +4 -1
- package/es/ContentTypeBadge/ContentTypeBadge.js +2 -0
- package/es/TreeStructure/TreeStructure.js +292 -181
- package/es/TreeStructure/helperFunctions.js +0 -3
- package/es/TreeStructure/index.js +1 -2
- package/es/index.js +0 -1
- package/es/locale/messages-en.js +1 -1
- package/es/locale/messages-nb.js +1 -1
- package/es/locale/messages-nn.js +1 -1
- package/es/locale/messages-se.js +1 -1
- package/es/locale/messages-sma.js +1 -1
- package/es/styles.css +24 -4
- package/lib/Article/Article.js +4 -1
- package/lib/ContentTypeBadge/ContentTypeBadge.js +2 -0
- package/lib/TreeStructure/TreeStructure.d.ts +7 -6
- package/lib/TreeStructure/TreeStructure.js +293 -180
- package/lib/TreeStructure/helperFunctions.d.ts +0 -2
- package/lib/TreeStructure/helperFunctions.js +2 -6
- package/lib/TreeStructure/index.d.ts +1 -2
- package/lib/TreeStructure/index.js +2 -3
- package/lib/TreeStructure/types.d.ts +4 -22
- package/lib/index.d.ts +0 -2
- package/lib/index.js +0 -7
- package/lib/locale/messages-en.js +1 -1
- package/lib/locale/messages-nb.js +1 -1
- package/lib/locale/messages-nn.js +1 -1
- package/lib/locale/messages-se.js +1 -1
- package/lib/locale/messages-sma.js +1 -1
- package/lib/styles.css +24 -4
- package/package.json +7 -8
- package/src/Article/Article.tsx +4 -1
- package/src/ContentTypeBadge/ContentTypeBadge.tsx +2 -0
- package/src/TreeStructure/TreeStructure.stories.tsx +38 -68
- package/src/TreeStructure/TreeStructure.tsx +307 -194
- package/src/TreeStructure/helperFunctions.ts +0 -5
- package/src/TreeStructure/index.ts +1 -2
- package/src/TreeStructure/types.ts +4 -25
- package/src/index.ts +0 -3
- package/src/locale/messages-en.ts +1 -1
- package/src/locale/messages-nb.ts +1 -1
- package/src/locale/messages-nn.ts +1 -1
- package/src/locale/messages-se.ts +1 -1
- package/src/locale/messages-sma.ts +1 -1
- package/es/ProgrammeCard/ProgrammeCard.js +0 -51
- package/es/ProgrammeCard/index.js +0 -9
- package/es/TreeStructure/AddFolderButton.js +0 -80
- package/es/TreeStructure/ComboboxButton.js +0 -127
- package/es/TreeStructure/FolderItem.js +0 -225
- package/es/TreeStructure/FolderItems.js +0 -95
- package/es/TreeStructure/arrowNavigation.js +0 -35
- package/lib/ProgrammeCard/ProgrammeCard.d.ts +0 -23
- package/lib/ProgrammeCard/ProgrammeCard.js +0 -58
- package/lib/ProgrammeCard/index.d.ts +0 -9
- package/lib/ProgrammeCard/index.js +0 -13
- package/lib/TreeStructure/AddFolderButton.d.ts +0 -17
- package/lib/TreeStructure/AddFolderButton.js +0 -85
- package/lib/TreeStructure/ComboboxButton.d.ts +0 -27
- package/lib/TreeStructure/ComboboxButton.js +0 -134
- package/lib/TreeStructure/FolderItem.d.ts +0 -17
- package/lib/TreeStructure/FolderItem.js +0 -230
- package/lib/TreeStructure/FolderItems.d.ts +0 -22
- package/lib/TreeStructure/FolderItems.js +0 -100
- package/lib/TreeStructure/arrowNavigation.d.ts +0 -10
- package/lib/TreeStructure/arrowNavigation.js +0 -42
- package/src/ProgrammeCard/ProgrammeCard.stories.tsx +0 -35
- package/src/ProgrammeCard/ProgrammeCard.tsx +0 -78
- package/src/ProgrammeCard/index.tsx +0 -10
- package/src/TreeStructure/AddFolderButton.tsx +0 -79
- package/src/TreeStructure/ComboboxButton.tsx +0 -172
- package/src/TreeStructure/FolderItem.tsx +0 -307
- package/src/TreeStructure/FolderItems.tsx +0 -121
- package/src/TreeStructure/arrowNavigation.ts +0 -54
|
@@ -1,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;
|