decap-cms-core 3.6.3 → 3.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (137) hide show
  1. package/dist/decap-cms-core.js +25 -25
  2. package/dist/decap-cms-core.js.LICENSE.txt +14 -8
  3. package/dist/decap-cms-core.js.map +1 -1
  4. package/dist/esm/actions/config.js +57 -49
  5. package/dist/esm/actions/editorialWorkflow.js +4 -4
  6. package/dist/esm/actions/entries.js +8 -14
  7. package/dist/esm/actions/mediaLibrary.js +6 -11
  8. package/dist/esm/actions/search.js +2 -2
  9. package/dist/esm/actions/status.js +2 -8
  10. package/dist/esm/backend.js +70 -79
  11. package/dist/esm/bootstrap.js +3 -2
  12. package/dist/esm/components/App/App.js +28 -34
  13. package/dist/esm/components/App/Header.js +32 -39
  14. package/dist/esm/components/Collection/Collection.js +45 -48
  15. package/dist/esm/components/Collection/CollectionSearch.js +76 -81
  16. package/dist/esm/components/Collection/CollectionTop.js +1 -2
  17. package/dist/esm/components/Collection/Entries/Entries.js +2 -4
  18. package/dist/esm/components/Collection/Entries/EntriesCollection.js +25 -29
  19. package/dist/esm/components/Collection/Entries/EntriesSearch.js +34 -38
  20. package/dist/esm/components/Collection/Entries/EntryCard.js +8 -13
  21. package/dist/esm/components/Collection/Entries/EntryListing.js +72 -76
  22. package/dist/esm/components/Collection/FilterControl.js +1 -1
  23. package/dist/esm/components/Collection/GroupControl.js +1 -1
  24. package/dist/esm/components/Collection/NestedCollection.js +50 -53
  25. package/dist/esm/components/Collection/Sidebar.js +35 -38
  26. package/dist/esm/components/Collection/SortControl.js +3 -3
  27. package/dist/esm/components/Collection/ViewStyleControl.js +1 -2
  28. package/dist/esm/components/Editor/Editor.js +197 -201
  29. package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +79 -87
  30. package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +75 -86
  31. package/dist/esm/components/Editor/EditorControlPane/Widget.js +226 -228
  32. package/dist/esm/components/Editor/EditorInterface.js +69 -80
  33. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +1 -2
  34. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +20 -28
  35. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +163 -161
  36. package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +4 -8
  37. package/dist/esm/components/Editor/EditorToolbar.js +335 -347
  38. package/dist/esm/components/Editor/withWorkflow.js +5 -6
  39. package/dist/esm/components/MediaLibrary/MediaLibrary.js +304 -294
  40. package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +40 -46
  41. package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +1 -2
  42. package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +8 -13
  43. package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +3 -3
  44. package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +1 -2
  45. package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +3 -6
  46. package/dist/esm/components/UI/DragDrop.js +15 -23
  47. package/dist/esm/components/UI/ErrorBoundary.js +23 -25
  48. package/dist/esm/components/UI/Modal.js +10 -12
  49. package/dist/esm/components/UI/Notifications.js +4 -8
  50. package/dist/esm/components/UI/SettingsDropdown.js +4 -8
  51. package/dist/esm/components/Workflow/Workflow.js +19 -20
  52. package/dist/esm/components/Workflow/WorkflowCard.js +2 -4
  53. package/dist/esm/components/Workflow/WorkflowList.js +105 -113
  54. package/dist/esm/constants/configSchema.js +18 -16
  55. package/dist/esm/formats/formats.js +11 -12
  56. package/dist/esm/formats/frontmatter.js +17 -21
  57. package/dist/esm/formats/toml.js +2 -2
  58. package/dist/esm/formats/yaml.js +2 -6
  59. package/dist/esm/index.js +3 -7
  60. package/dist/esm/integrations/providers/algolia/implementation.js +12 -14
  61. package/dist/esm/integrations/providers/assetStore/implementation.js +10 -12
  62. package/dist/esm/lib/formatters.js +13 -17
  63. package/dist/esm/lib/i18n.js +35 -33
  64. package/dist/esm/lib/phrases.js +2 -2
  65. package/dist/esm/lib/polyfill.js +8 -0
  66. package/dist/esm/lib/registry.js +35 -35
  67. package/dist/esm/lib/serializeEntryValues.js +3 -3
  68. package/dist/esm/lib/stega.js +142 -0
  69. package/dist/esm/lib/urlHelper.js +16 -18
  70. package/dist/esm/mediaLibrary.js +3 -4
  71. package/dist/esm/reducers/collections.js +26 -42
  72. package/dist/esm/reducers/combinedReducer.js +3 -6
  73. package/dist/esm/reducers/config.js +3 -7
  74. package/dist/esm/reducers/editorialWorkflow.js +5 -9
  75. package/dist/esm/reducers/entries.js +33 -35
  76. package/dist/esm/reducers/entryDraft.js +2 -2
  77. package/dist/esm/reducers/integrations.js +8 -14
  78. package/dist/esm/reducers/mediaLibrary.js +18 -20
  79. package/dist/esm/reducers/notifications.js +4 -8
  80. package/dist/esm/types/immutable.js +7 -1
  81. package/dist/esm/valueObjects/AssetProxy.js +1 -9
  82. package/dist/esm/valueObjects/EditorComponent.js +18 -25
  83. package/dist/esm/valueObjects/Entry.js +2 -2
  84. package/index.d.ts +2 -0
  85. package/package.json +14 -11
  86. package/src/actions/__tests__/config.spec.js +3 -3
  87. package/src/actions/config.ts +3 -1
  88. package/src/actions/editorialWorkflow.ts +1 -1
  89. package/src/actions/entries.ts +1 -1
  90. package/src/actions/search.ts +1 -1
  91. package/src/backend.ts +8 -1
  92. package/src/bootstrap.js +1 -0
  93. package/src/components/App/App.js +5 -0
  94. package/src/components/App/Header.js +3 -0
  95. package/src/components/Collection/Collection.js +5 -0
  96. package/src/components/Collection/CollectionSearch.js +5 -0
  97. package/src/components/Collection/Entries/EntriesCollection.js +4 -1
  98. package/src/components/Collection/Entries/EntriesSearch.js +4 -1
  99. package/src/components/Collection/Entries/EntryListing.js +5 -0
  100. package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +0 -4
  101. package/src/components/Collection/NestedCollection.js +6 -1
  102. package/src/components/Collection/Sidebar.js +5 -0
  103. package/src/components/Editor/Editor.js +4 -1
  104. package/src/components/Editor/EditorControlPane/EditorControl.js +7 -1
  105. package/src/components/Editor/EditorControlPane/Widget.js +5 -0
  106. package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +1 -1
  107. package/src/components/Editor/EditorToolbar.js +3 -0
  108. package/src/components/Editor/__tests__/Editor.spec.js +3 -4
  109. package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +5 -5
  110. package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +708 -393
  111. package/src/components/MediaLibrary/MediaLibrary.js +5 -1
  112. package/src/components/MediaLibrary/MediaLibraryModal.js +1 -1
  113. package/src/components/UI/ErrorBoundary.js +6 -1
  114. package/src/components/UI/Modal.js +3 -0
  115. package/src/components/Workflow/Workflow.js +3 -0
  116. package/src/components/Workflow/WorkflowList.js +5 -0
  117. package/src/constants/__tests__/configSchema.spec.js +1 -1
  118. package/src/formats/formats.ts +1 -1
  119. package/src/formats/toml.ts +2 -2
  120. package/src/integrations/providers/algolia/implementation.js +2 -2
  121. package/src/integrations/providers/assetStore/implementation.js +2 -1
  122. package/src/lib/formatters.ts +4 -1
  123. package/src/lib/i18n.ts +3 -1
  124. package/src/lib/phrases.js +1 -1
  125. package/src/lib/polyfill.js +9 -0
  126. package/src/lib/serializeEntryValues.js +1 -1
  127. package/src/lib/stega.ts +145 -0
  128. package/src/lib/urlHelper.ts +4 -1
  129. package/src/mediaLibrary.ts +1 -1
  130. package/src/reducers/collections.ts +2 -1
  131. package/src/reducers/editorialWorkflow.ts +1 -1
  132. package/src/reducers/entries.ts +6 -1
  133. package/src/reducers/entryDraft.js +1 -1
  134. package/src/types/immutable.ts +10 -0
  135. package/src/types/redux.ts +2 -0
  136. package/src/valueObjects/EditorComponent.js +1 -1
  137. package/src/valueObjects/Entry.ts +1 -1
@@ -9,7 +9,7 @@ function GroupControl({
9
9
  onGroupClick,
10
10
  group
11
11
  }) {
12
- const hasActiveGroup = group === null || group === void 0 ? void 0 : group.valueSeq().toJS().some(f => f.active === true);
12
+ const hasActiveGroup = group?.valueSeq().toJS().some(f => f.active === true);
13
13
  return ___EmotionJSX(Dropdown, {
14
14
  renderButton: () => {
15
15
  return ___EmotionJSX(ControlButton, {
@@ -1,10 +1,4 @@
1
1
  import _styled from "@emotion/styled/base";
2
- import _sortBy from "lodash/sortBy";
3
- function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
4
- function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
5
- function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
6
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); }
7
- function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
8
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)."; }
9
3
  import React from 'react';
10
4
  import { List } from 'immutable';
@@ -16,6 +10,7 @@ import { stringTemplate } from 'decap-cms-lib-widgets';
16
10
  import { Icon, colors, components } from 'decap-cms-ui-default';
17
11
  import PropTypes from 'prop-types';
18
12
  import ImmutablePropTypes from 'react-immutable-proptypes';
13
+ import sortBy from 'lodash/sortBy';
19
14
  import { selectEntries } from '../../reducers/entries';
20
15
  import { selectEntryCollectionTitle } from '../../reducers/collections';
21
16
  import { jsx as ___EmotionJSX } from "@emotion/react";
@@ -30,8 +25,7 @@ const NodeTitleContainer = /*#__PURE__*/_styled("div", {
30
25
  styles: "display:flex;justify-content:center;align-items:center"
31
26
  } : {
32
27
  name: "1vcob1d",
33
- styles: "display:flex;justify-content:center;align-items:center",
34
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAkBqC","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { sortBy } from 'lodash';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */",
28
+ styles: "display:flex;justify-content:center;align-items:center/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAkBqC","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport sortBy from 'lodash/sortBy';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */",
35
29
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
36
30
  });
37
31
  const NodeTitle = /*#__PURE__*/_styled("div", {
@@ -42,8 +36,7 @@ const NodeTitle = /*#__PURE__*/_styled("div", {
42
36
  styles: "margin-right:4px"
43
37
  } : {
44
38
  name: "qamjgr",
45
- styles: "margin-right:4px",
46
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAwB4B","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { sortBy } from 'lodash';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */",
39
+ styles: "margin-right:4px/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAwB4B","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport sortBy from 'lodash/sortBy';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */",
47
40
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
48
41
  });
49
42
  const Caret = /*#__PURE__*/_styled("div", {
@@ -54,29 +47,26 @@ const Caret = /*#__PURE__*/_styled("div", {
54
47
  styles: "position:relative;top:2px"
55
48
  } : {
56
49
  name: "1lo1a34",
57
- styles: "position:relative;top:2px",
58
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AA4BwB","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { sortBy } from 'lodash';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */",
50
+ styles: "position:relative;top:2px/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AA4BwB","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport sortBy from 'lodash/sortBy';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */",
59
51
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
60
52
  });
61
53
  const CaretDown = /*#__PURE__*/_styled(Caret, {
62
54
  target: "eh84wlq2",
63
55
  label: "CaretDown"
64
- })(components.caretDown, ";color:currentColor;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAiC+B","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { sortBy } from 'lodash';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */"));
56
+ })(components.caretDown, ";color:currentColor;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAiC+B","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport sortBy from 'lodash/sortBy';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */"));
65
57
  const CaretRight = /*#__PURE__*/_styled(Caret, {
66
58
  target: "eh84wlq1",
67
59
  label: "CaretRight"
68
- })(components.caretRight, ";color:currentColor;left:2px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAsCgC","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { sortBy } from 'lodash';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */"));
60
+ })(components.caretRight, ";color:currentColor;left:2px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AAsCgC","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport sortBy from 'lodash/sortBy';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */"));
69
61
  const TreeNavLink = /*#__PURE__*/_styled(NavLink, {
70
62
  target: "eh84wlq0",
71
63
  label: "TreeNavLink"
72
- })("display:flex;font-size:14px;font-weight:500;align-items:center;padding:8px;padding-left:", props => props.depth * 16 + 18, "px;border-left:2px solid #fff;", Icon, "{margin-right:4px;flex-shrink:0;}", props => /*#__PURE__*/css("&:hover,&:active,&.", props.activeClassName, "{color:", colors.active, ";background-color:", colors.activeBackground, ";border-left-color:#4863c6;};label:TreeNavLink;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AA0DgB","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { sortBy } from 'lodash';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */")), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AA4CmC","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { sortBy } from 'lodash';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */"));
64
+ })("display:flex;font-size:14px;font-weight:500;align-items:center;padding:8px;padding-left:", props => props.depth * 16 + 18, "px;border-left:2px solid #fff;", Icon, "{margin-right:4px;flex-shrink:0;}", props => /*#__PURE__*/css("&:hover,&:active,&.", props.activeClassName, "{color:", colors.active, ";background-color:", colors.activeBackground, ";border-left-color:#4863c6;};label:TreeNavLink;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AA0DgB","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport sortBy from 'lodash/sortBy';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */")), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Collection/NestedCollection.js"],"names":[],"mappings":"AA4CmC","file":"../../../../src/components/Collection/NestedCollection.js","sourcesContent":["import React from 'react';\nimport { List } from 'immutable';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { connect } from 'react-redux';\nimport { NavLink } from 'react-router-dom';\nimport { dirname, sep } from 'path';\nimport { stringTemplate } from 'decap-cms-lib-widgets';\nimport { Icon, colors, components } from 'decap-cms-ui-default';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport sortBy from 'lodash/sortBy';\n\nimport { selectEntries } from '../../reducers/entries';\nimport { selectEntryCollectionTitle } from '../../reducers/collections';\n\nconst { addFileTemplateFields } = stringTemplate;\n\nconst NodeTitleContainer = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst NodeTitle = styled.div`\n  margin-right: 4px;\n`;\n\nconst Caret = styled.div`\n  position: relative;\n  top: 2px;\n`;\n\nconst CaretDown = styled(Caret)`\n  ${components.caretDown};\n  color: currentColor;\n`;\n\nconst CaretRight = styled(Caret)`\n  ${components.caretRight};\n  color: currentColor;\n  left: 2px;\n`;\n\nconst TreeNavLink = styled(NavLink)`\n  display: flex;\n  font-size: 14px;\n  font-weight: 500;\n  align-items: center;\n  padding: 8px;\n  padding-left: ${props => props.depth * 16 + 18}px;\n  border-left: 2px solid #fff;\n\n  ${Icon} {\n    margin-right: 4px;\n    flex-shrink: 0;\n  }\n\n  ${props => css`\n    &:hover,\n    &:active,\n    &.${props.activeClassName} {\n      color: ${colors.active};\n      background-color: ${colors.activeBackground};\n      border-left-color: #4863c6;\n    }\n  `};\n`;\n\nfunction getNodeTitle(node) {\n  const title = node.isRoot\n    ? node.title\n    : node.children.find(c => !c.isDir && c.title)?.title || node.title;\n  return title;\n}\n\nfunction TreeNode(props) {\n  const { collection, treeData, depth = 0, onToggle } = props;\n  const collectionName = collection.get('name');\n\n  const sortedData = sortBy(treeData, getNodeTitle);\n  const subfolders = collection.get('nested')?.get('subfolders') !== false;\n  return sortedData.map(node => {\n    const leaf =\n      depth > 0 &&\n      (subfolders\n        ? node.children.length <= 1 && !node.children[0]?.isDir\n        : node.children.length === 0);\n    if (leaf) {\n      return null;\n    }\n    let to = `/collections/${collectionName}`;\n    if (depth > 0) {\n      to = `${to}/filter${node.path}`;\n    }\n    const title = getNodeTitle(node);\n\n    const hasChildren =\n      depth === 0 ||\n      (subfolders\n        ? node.children.some(c => c.children.some(c => c.isDir))\n        : node.children.some(c => c.isDir));\n\n    return (\n      <React.Fragment key={node.path}>\n        <TreeNavLink\n          exact\n          to={to}\n          activeClassName=\"sidebar-active\"\n          onClick={() => onToggle({ node, expanded: !node.expanded })}\n          depth={depth}\n          data-testid={node.path}\n        >\n          <Icon type=\"write\" />\n          <NodeTitleContainer>\n            <NodeTitle>{title}</NodeTitle>\n            {hasChildren && (node.expanded ? <CaretDown /> : <CaretRight />)}\n          </NodeTitleContainer>\n        </TreeNavLink>\n        {node.expanded && (\n          <TreeNode\n            collection={collection}\n            depth={depth + 1}\n            treeData={node.children}\n            onToggle={onToggle}\n          />\n        )}\n      </React.Fragment>\n    );\n  });\n}\n\nTreeNode.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  depth: PropTypes.number,\n  treeData: PropTypes.array.isRequired,\n  onToggle: PropTypes.func.isRequired,\n};\n\nexport function walk(treeData, callback) {\n  function traverse(children) {\n    for (const child of children) {\n      callback(child);\n      traverse(child.children);\n    }\n  }\n\n  return traverse(treeData);\n}\n\nexport function getTreeData(collection, entries) {\n  const collectionFolder = collection.get('folder');\n  const rootFolder = '/';\n  const entriesObj = entries\n    .toJS()\n    .map(e => ({ ...e, path: e.path.slice(collectionFolder.length) }));\n\n  const dirs = entriesObj.reduce((acc, entry) => {\n    let dir = dirname(entry.path);\n    while (!acc[dir] && dir && dir !== rootFolder) {\n      const parts = dir.split(sep);\n      acc[dir] = parts.pop();\n      dir = parts.length && parts.join(sep);\n    }\n    return acc;\n  }, {});\n\n  if (collection.getIn(['nested', 'summary'])) {\n    collection = collection.set('summary', collection.getIn(['nested', 'summary']));\n  } else {\n    collection = collection.delete('summary');\n  }\n\n  const flatData = [\n    {\n      title: collection.get('label'),\n      path: rootFolder,\n      isDir: true,\n      isRoot: true,\n    },\n    ...Object.entries(dirs).map(([key, value]) => ({\n      title: value,\n      path: key,\n      isDir: true,\n      isRoot: false,\n    })),\n    ...entriesObj.map((e, index) => {\n      let entryMap = entries.get(index);\n      entryMap = entryMap.set(\n        'data',\n        addFileTemplateFields(entryMap.get('path'), entryMap.get('data')),\n      );\n      const title = selectEntryCollectionTitle(collection, entryMap);\n      return {\n        ...e,\n        title,\n        isDir: false,\n        isRoot: false,\n      };\n    }),\n  ];\n\n  const parentsToChildren = flatData.reduce((acc, node) => {\n    const parent = node.path === rootFolder ? '' : dirname(node.path);\n    if (acc[parent]) {\n      acc[parent].push(node);\n    } else {\n      acc[parent] = [node];\n    }\n    return acc;\n  }, {});\n\n  function reducer(acc, value) {\n    const node = value;\n    let children = [];\n    if (parentsToChildren[node.path]) {\n      children = parentsToChildren[node.path].reduce(reducer, []);\n    }\n\n    acc.push({ ...node, children });\n    return acc;\n  }\n\n  const treeData = parentsToChildren[''].reduce(reducer, []);\n\n  return treeData;\n}\n\nexport function updateNode(treeData, node, callback) {\n  let stop = false;\n\n  function updater(nodes) {\n    if (stop) {\n      return nodes;\n    }\n    for (let i = 0; i < nodes.length; i++) {\n      if (nodes[i].path === node.path) {\n        nodes[i] = callback(node);\n        stop = true;\n        return nodes;\n      }\n    }\n    nodes.forEach(node => updater(node.children));\n    return nodes;\n  }\n\n  return updater([...treeData]);\n}\n\nexport class NestedCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    entries: ImmutablePropTypes.list.isRequired,\n    filterTerm: PropTypes.string,\n  };\n\n  constructor(props) {\n    super(props);\n    this.state = {\n      treeData: getTreeData(this.props.collection, this.props.entries),\n      selected: null,\n      useFilter: true,\n    };\n  }\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entries, filterTerm } = this.props;\n    if (\n      collection !== prevProps.collection ||\n      entries !== prevProps.entries ||\n      filterTerm !== prevProps.filterTerm\n    ) {\n      const expanded = {};\n      walk(this.state.treeData, node => {\n        if (node.expanded) {\n          expanded[node.path] = true;\n        }\n      });\n      const treeData = getTreeData(collection, entries);\n\n      const path = `/${filterTerm}`;\n      walk(treeData, node => {\n        if (expanded[node.path] || (this.state.useFilter && path.startsWith(node.path))) {\n          node.expanded = true;\n        }\n      });\n      this.setState({ treeData });\n    }\n  }\n\n  onToggle = ({ node, expanded }) => {\n    if (!this.state.selected || this.state.selected.path === node.path || expanded) {\n      const treeData = updateNode(this.state.treeData, node, node => ({\n        ...node,\n        expanded,\n      }));\n      this.setState({ treeData, selected: node, useFilter: false });\n    } else {\n      // don't collapse non selected nodes when clicked\n      this.setState({ selected: node, useFilter: false });\n    }\n  };\n\n  render() {\n    const { treeData } = this.state;\n    const { collection } = this.props;\n\n    return <TreeNode collection={collection} treeData={treeData} onToggle={this.onToggle} />;\n  }\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection } = ownProps;\n  const entries = selectEntries(state.entries, collection) || List();\n  return { entries };\n}\n\nexport default connect(mapStateToProps, null)(NestedCollection);\n"]} */"));
73
65
  function getNodeTitle(node) {
74
- var _node$children$find;
75
- const title = node.isRoot ? node.title : ((_node$children$find = node.children.find(c => !c.isDir && c.title)) === null || _node$children$find === void 0 ? void 0 : _node$children$find.title) || node.title;
66
+ const title = node.isRoot ? node.title : node.children.find(c => !c.isDir && c.title)?.title || node.title;
76
67
  return title;
77
68
  }
78
69
  function TreeNode(props) {
79
- var _collection$get;
80
70
  const {
81
71
  collection,
82
72
  treeData,
@@ -84,11 +74,10 @@ function TreeNode(props) {
84
74
  onToggle
85
75
  } = props;
86
76
  const collectionName = collection.get('name');
87
- const sortedData = _sortBy(treeData, getNodeTitle);
88
- const subfolders = ((_collection$get = collection.get('nested')) === null || _collection$get === void 0 ? void 0 : _collection$get.get('subfolders')) !== false;
77
+ const sortedData = sortBy(treeData, getNodeTitle);
78
+ const subfolders = collection.get('nested')?.get('subfolders') !== false;
89
79
  return sortedData.map(node => {
90
- var _node$children$;
91
- const leaf = depth > 0 && (subfolders ? node.children.length <= 1 && !((_node$children$ = node.children[0]) !== null && _node$children$ !== void 0 && _node$children$.isDir) : node.children.length === 0);
80
+ const leaf = depth > 0 && (subfolders ? node.children.length <= 1 && !node.children[0]?.isDir : node.children.length === 0);
92
81
  if (leaf) {
93
82
  return null;
94
83
  }
@@ -138,7 +127,8 @@ export function walk(treeData, callback) {
138
127
  export function getTreeData(collection, entries) {
139
128
  const collectionFolder = collection.get('folder');
140
129
  const rootFolder = '/';
141
- const entriesObj = entries.toJS().map(e => _objectSpread(_objectSpread({}, e), {}, {
130
+ const entriesObj = entries.toJS().map(e => ({
131
+ ...e,
142
132
  path: e.path.slice(collectionFolder.length)
143
133
  }));
144
134
  const dirs = entriesObj.reduce((acc, entry) => {
@@ -169,11 +159,12 @@ export function getTreeData(collection, entries) {
169
159
  let entryMap = entries.get(index);
170
160
  entryMap = entryMap.set('data', addFileTemplateFields(entryMap.get('path'), entryMap.get('data')));
171
161
  const title = selectEntryCollectionTitle(collection, entryMap);
172
- return _objectSpread(_objectSpread({}, e), {}, {
162
+ return {
163
+ ...e,
173
164
  title,
174
165
  isDir: false,
175
166
  isRoot: false
176
- });
167
+ };
177
168
  })];
178
169
  const parentsToChildren = flatData.reduce((acc, node) => {
179
170
  const parent = node.path === rootFolder ? '' : dirname(node.path);
@@ -190,9 +181,10 @@ export function getTreeData(collection, entries) {
190
181
  if (parentsToChildren[node.path]) {
191
182
  children = parentsToChildren[node.path].reduce(reducer, []);
192
183
  }
193
- acc.push(_objectSpread(_objectSpread({}, node), {}, {
184
+ acc.push({
185
+ ...node,
194
186
  children
195
- }));
187
+ });
196
188
  return acc;
197
189
  }
198
190
  const treeData = parentsToChildren[''].reduce(reducer, []);
@@ -217,35 +209,23 @@ export function updateNode(treeData, node, callback) {
217
209
  return updater([...treeData]);
218
210
  }
219
211
  export class NestedCollection extends React.Component {
212
+ static propTypes = {
213
+ collection: ImmutablePropTypes.map.isRequired,
214
+ entries: ImmutablePropTypes.list.isRequired,
215
+ filterTerm: PropTypes.string
216
+ };
220
217
  constructor(props) {
221
218
  super(props);
222
- _defineProperty(this, "onToggle", ({
223
- node,
224
- expanded
225
- }) => {
226
- if (!this.state.selected || this.state.selected.path === node.path || expanded) {
227
- const treeData = updateNode(this.state.treeData, node, node => _objectSpread(_objectSpread({}, node), {}, {
228
- expanded
229
- }));
230
- this.setState({
231
- treeData,
232
- selected: node,
233
- useFilter: false
234
- });
235
- } else {
236
- // don't collapse non selected nodes when clicked
237
- this.setState({
238
- selected: node,
239
- useFilter: false
240
- });
241
- }
242
- });
243
219
  this.state = {
244
220
  treeData: getTreeData(this.props.collection, this.props.entries),
245
221
  selected: null,
246
222
  useFilter: true
247
223
  };
248
224
  }
225
+ componentDidMount() {
226
+ // Manually validate PropTypes - React 19 breaking change
227
+ PropTypes.checkPropTypes(NestedCollection.propTypes, this.props, 'prop', 'NestedCollection');
228
+ }
249
229
  componentDidUpdate(prevProps) {
250
230
  const {
251
231
  collection,
@@ -271,6 +251,28 @@ export class NestedCollection extends React.Component {
271
251
  });
272
252
  }
273
253
  }
254
+ onToggle = ({
255
+ node,
256
+ expanded
257
+ }) => {
258
+ if (!this.state.selected || this.state.selected.path === node.path || expanded) {
259
+ const treeData = updateNode(this.state.treeData, node, node => ({
260
+ ...node,
261
+ expanded
262
+ }));
263
+ this.setState({
264
+ treeData,
265
+ selected: node,
266
+ useFilter: false
267
+ });
268
+ } else {
269
+ // don't collapse non selected nodes when clicked
270
+ this.setState({
271
+ selected: node,
272
+ useFilter: false
273
+ });
274
+ }
275
+ };
274
276
  render() {
275
277
  const {
276
278
  treeData
@@ -285,11 +287,6 @@ export class NestedCollection extends React.Component {
285
287
  });
286
288
  }
287
289
  }
288
- _defineProperty(NestedCollection, "propTypes", {
289
- collection: ImmutablePropTypes.map.isRequired,
290
- entries: ImmutablePropTypes.list.isRequired,
291
- filterTerm: PropTypes.string
292
- });
293
290
  function mapStateToProps(state, ownProps) {
294
291
  const {
295
292
  collection