decap-cms-core 3.5.0 → 3.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/decap-cms-core.js +9 -9
- package/dist/decap-cms-core.js.map +1 -1
- package/dist/esm/actions/auth.js +22 -37
- package/dist/esm/actions/collections.js +9 -17
- package/dist/esm/actions/config.js +58 -74
- package/dist/esm/actions/deploys.js +10 -17
- package/dist/esm/actions/editorialWorkflow.js +87 -101
- package/dist/esm/actions/entries.js +161 -211
- package/dist/esm/actions/media.js +31 -46
- package/dist/esm/actions/mediaLibrary.js +94 -126
- package/dist/esm/actions/notifications.js +5 -13
- package/dist/esm/actions/search.js +30 -47
- package/dist/esm/actions/status.js +13 -23
- package/dist/esm/actions/waitUntil.js +4 -11
- package/dist/esm/backend.js +132 -148
- package/dist/esm/bootstrap.js +37 -44
- package/dist/esm/components/App/App.js +82 -89
- package/dist/esm/components/App/Header.js +46 -52
- package/dist/esm/components/App/NotFoundPage.js +11 -18
- package/dist/esm/components/Collection/Collection.js +55 -63
- package/dist/esm/components/Collection/CollectionControls.js +15 -22
- package/dist/esm/components/Collection/CollectionSearch.js +35 -42
- package/dist/esm/components/Collection/CollectionTop.js +23 -30
- package/dist/esm/components/Collection/ControlButton.js +10 -16
- package/dist/esm/components/Collection/Entries/Entries.js +24 -31
- package/dist/esm/components/Collection/Entries/EntriesCollection.js +52 -62
- package/dist/esm/components/Collection/Entries/EntriesSearch.js +26 -33
- package/dist/esm/components/Collection/Entries/EntryCard.js +38 -45
- package/dist/esm/components/Collection/Entries/EntryListing.js +24 -32
- package/dist/esm/components/Collection/FilterControl.js +9 -16
- package/dist/esm/components/Collection/GroupControl.js +9 -16
- package/dist/esm/components/Collection/NestedCollection.js +54 -65
- package/dist/esm/components/Collection/Sidebar.js +36 -43
- package/dist/esm/components/Collection/SortControl.js +19 -26
- package/dist/esm/components/Collection/ViewStyleControl.js +17 -24
- package/dist/esm/components/Editor/Editor.js +100 -108
- package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +105 -112
- package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +68 -62
- package/dist/esm/components/Editor/EditorControlPane/Widget.js +87 -73
- package/dist/esm/components/Editor/EditorInterface.js +95 -98
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +13 -21
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +64 -23
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +94 -78
- package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +9 -16
- package/dist/esm/components/Editor/EditorToolbar.js +133 -140
- package/dist/esm/components/Editor/withWorkflow.js +15 -22
- package/dist/esm/components/EditorWidgets/Unknown/UnknownControl.js +9 -16
- package/dist/esm/components/EditorWidgets/Unknown/UnknownPreview.js +9 -16
- package/dist/esm/components/EditorWidgets/index.js +4 -7
- package/dist/esm/components/MediaLibrary/EmptyMessage.js +12 -19
- package/dist/esm/components/MediaLibrary/MediaLibrary.js +55 -62
- package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +28 -35
- package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +36 -43
- package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +50 -57
- package/dist/esm/components/MediaLibrary/MediaLibraryHeader.js +16 -23
- package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +59 -64
- package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +18 -25
- package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +39 -46
- package/dist/esm/components/UI/DragDrop.js +21 -30
- package/dist/esm/components/UI/ErrorBoundary.js +35 -43
- package/dist/esm/components/UI/FileUploadButton.js +11 -18
- package/dist/esm/components/UI/Modal.js +19 -26
- package/dist/esm/components/UI/Notifications.js +21 -28
- package/dist/esm/components/UI/SettingsDropdown.js +28 -34
- package/dist/esm/components/UI/index.js +6 -60
- package/dist/esm/components/Workflow/Workflow.js +52 -61
- package/dist/esm/components/Workflow/WorkflowCard.js +45 -51
- package/dist/esm/components/Workflow/WorkflowList.js +43 -49
- package/dist/esm/constants/collectionTypes.js +2 -8
- package/dist/esm/constants/collectionViews.js +2 -8
- package/dist/esm/constants/commitProps.js +2 -8
- package/dist/esm/constants/configSchema.js +20 -27
- package/dist/esm/constants/fieldInference.js +8 -15
- package/dist/esm/constants/publishModes.js +6 -11
- package/dist/esm/constants/validationErrorTypes.js +1 -7
- package/dist/esm/formats/formats.js +32 -41
- package/dist/esm/formats/frontmatter.js +18 -30
- package/dist/esm/formats/helpers.js +1 -7
- package/dist/esm/formats/json.js +1 -7
- package/dist/esm/formats/toml.js +11 -18
- package/dist/esm/formats/yaml.js +7 -14
- package/dist/esm/index.js +5 -12
- package/dist/esm/integrations/index.js +8 -16
- package/dist/esm/integrations/providers/algolia/implementation.js +14 -22
- package/dist/esm/integrations/providers/assetStore/implementation.js +10 -18
- package/dist/esm/lib/consoleError.js +1 -7
- package/dist/esm/lib/formatters.js +34 -47
- package/dist/esm/lib/i18n.js +37 -66
- package/dist/esm/lib/phrases.js +4 -11
- package/dist/esm/lib/registry.js +40 -75
- package/dist/esm/lib/serializeEntryValues.js +11 -18
- package/dist/esm/lib/textHelper.js +1 -7
- package/dist/esm/lib/urlHelper.js +28 -43
- package/dist/esm/mediaLibrary.js +12 -16
- package/dist/esm/reducers/auth.js +10 -16
- package/dist/esm/reducers/collections.js +70 -102
- package/dist/esm/reducers/combinedReducer.js +4 -11
- package/dist/esm/reducers/config.js +11 -19
- package/dist/esm/reducers/cursors.js +12 -18
- package/dist/esm/reducers/deploys.js +8 -15
- package/dist/esm/reducers/editorialWorkflow.js +37 -47
- package/dist/esm/reducers/entries.js +107 -132
- package/dist/esm/reducers/entryDraft.js +64 -72
- package/dist/esm/reducers/globalUI.js +5 -11
- package/dist/esm/reducers/index.js +43 -64
- package/dist/esm/reducers/integrations.js +8 -16
- package/dist/esm/reducers/mediaLibrary.js +43 -52
- package/dist/esm/reducers/medias.js +11 -18
- package/dist/esm/reducers/notifications.js +9 -15
- package/dist/esm/reducers/search.js +12 -18
- package/dist/esm/reducers/status.js +7 -13
- package/dist/esm/redux/index.js +7 -13
- package/dist/esm/redux/middleware/waitUntilAction.js +3 -10
- package/dist/esm/routing/history.js +7 -15
- package/dist/esm/types/diacritics.d.js +0 -1
- package/dist/esm/types/global.d.js +1 -5
- package/dist/esm/types/immutable.js +1 -5
- package/dist/esm/types/redux.js +7 -8
- package/dist/esm/types/tomlify-j0.4.d.js +0 -1
- package/dist/esm/valueObjects/AssetProxy.js +2 -10
- package/dist/esm/valueObjects/EditorComponent.js +5 -12
- package/dist/esm/valueObjects/Entry.js +3 -10
- package/package.json +3 -2
- package/src/components/Collection/Entries/EntriesCollection.js +7 -8
- package/src/components/Collection/Entries/__tests__/EntriesCollection.spec.js +3 -5
- package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +8 -8
- package/src/components/Collection/NestedCollection.js +2 -2
- package/src/components/Collection/__tests__/__snapshots__/NestedCollection.spec.js.snap +68 -0
- package/src/components/Editor/EditorControlPane/EditorControl.js +0 -3
- package/src/components/Editor/EditorControlPane/EditorControlPane.js +21 -8
- package/src/components/Editor/EditorControlPane/Widget.js +22 -1
- package/src/components/Editor/EditorInterface.js +6 -1
- package/src/components/Editor/EditorPreviewPane/EditorPreviewContent.js +51 -11
- package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +33 -1
- package/dist/esm/actions/editorControl.js +0 -14
- package/dist/esm/reducers/editorComponent.js +0 -1
- package/dist/esm/reducers/editorControl.js +0 -17
|
@@ -1,37 +1,28 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
Object.defineProperty(exports, "__esModule", {
|
|
4
|
-
value: true
|
|
5
|
-
});
|
|
6
|
-
exports.default = exports.NestedCollection = void 0;
|
|
7
|
-
exports.getTreeData = getTreeData;
|
|
8
|
-
exports.updateNode = updateNode;
|
|
9
|
-
exports.walk = walk;
|
|
10
|
-
var _base = _interopRequireDefault(require("@emotion/styled/base"));
|
|
11
|
-
var _sortBy2 = _interopRequireDefault(require("lodash/sortBy"));
|
|
12
|
-
var _react = _interopRequireDefault(require("react"));
|
|
13
|
-
var _immutable = require("immutable");
|
|
14
|
-
var _react2 = require("@emotion/react");
|
|
15
|
-
var _reactRedux = require("react-redux");
|
|
16
|
-
var _reactRouterDom = require("react-router-dom");
|
|
17
|
-
var _path = require("path");
|
|
18
|
-
var _decapCmsLibWidgets = require("decap-cms-lib-widgets");
|
|
19
|
-
var _decapCmsUiDefault = require("decap-cms-ui-default");
|
|
20
|
-
var _propTypes = _interopRequireDefault(require("prop-types"));
|
|
21
|
-
var _reactImmutableProptypes = _interopRequireDefault(require("react-immutable-proptypes"));
|
|
22
|
-
var _entries = require("../../reducers/entries");
|
|
23
|
-
var _collections = require("../../reducers/collections");
|
|
24
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
1
|
+
import _styled from "@emotion/styled/base";
|
|
2
|
+
import _sortBy from "lodash/sortBy";
|
|
25
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; }
|
|
26
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; }
|
|
27
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; }
|
|
28
6
|
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); }
|
|
29
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); }
|
|
30
8
|
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
|
+
import React from 'react';
|
|
10
|
+
import { List } from 'immutable';
|
|
11
|
+
import { css } from '@emotion/react';
|
|
12
|
+
import { connect } from 'react-redux';
|
|
13
|
+
import { NavLink } from 'react-router-dom';
|
|
14
|
+
import { dirname, sep } from 'path';
|
|
15
|
+
import { stringTemplate } from 'decap-cms-lib-widgets';
|
|
16
|
+
import { Icon, colors, components } from 'decap-cms-ui-default';
|
|
17
|
+
import PropTypes from 'prop-types';
|
|
18
|
+
import ImmutablePropTypes from 'react-immutable-proptypes';
|
|
19
|
+
import { selectEntries } from '../../reducers/entries';
|
|
20
|
+
import { selectEntryCollectionTitle } from '../../reducers/collections';
|
|
21
|
+
import { jsx as ___EmotionJSX } from "@emotion/react";
|
|
31
22
|
const {
|
|
32
23
|
addFileTemplateFields
|
|
33
|
-
} =
|
|
34
|
-
const NodeTitleContainer = /*#__PURE__*/(
|
|
24
|
+
} = stringTemplate;
|
|
25
|
+
const NodeTitleContainer = /*#__PURE__*/_styled("div", {
|
|
35
26
|
target: "eh84wlq5",
|
|
36
27
|
label: "NodeTitleContainer"
|
|
37
28
|
})(process.env.NODE_ENV === "production" ? {
|
|
@@ -40,10 +31,10 @@ const NodeTitleContainer = /*#__PURE__*/(0, _base.default)("div", {
|
|
|
40
31
|
} : {
|
|
41
32
|
name: "1vcob1d",
|
|
42
33
|
styles: "display:flex;justify-content:center;align-items:center",
|
|
43
|
-
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  return sortedData.map(node => {\n    const leaf = node.children.length <= 1 && !node.children[0]?.isDir && depth > 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 = depth === 0 || node.children.some(c => c.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"]} */",
|
|
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  return sortedData.map(node => {\n    const leaf = node.children.length === 0 && depth > 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 = depth === 0 || 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"]} */",
|
|
44
35
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
45
36
|
});
|
|
46
|
-
const NodeTitle = /*#__PURE__*/(
|
|
37
|
+
const NodeTitle = /*#__PURE__*/_styled("div", {
|
|
47
38
|
target: "eh84wlq4",
|
|
48
39
|
label: "NodeTitle"
|
|
49
40
|
})(process.env.NODE_ENV === "production" ? {
|
|
@@ -52,10 +43,10 @@ const NodeTitle = /*#__PURE__*/(0, _base.default)("div", {
|
|
|
52
43
|
} : {
|
|
53
44
|
name: "qamjgr",
|
|
54
45
|
styles: "margin-right:4px",
|
|
55
|
-
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  return sortedData.map(node => {\n    const leaf = node.children.length <= 1 && !node.children[0]?.isDir && depth > 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 = depth === 0 || node.children.some(c => c.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"]} */",
|
|
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  return sortedData.map(node => {\n    const leaf = node.children.length === 0 && depth > 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 = depth === 0 || 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
47
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
57
48
|
});
|
|
58
|
-
const Caret = /*#__PURE__*/(
|
|
49
|
+
const Caret = /*#__PURE__*/_styled("div", {
|
|
59
50
|
target: "eh84wlq3",
|
|
60
51
|
label: "Caret"
|
|
61
52
|
})(process.env.NODE_ENV === "production" ? {
|
|
@@ -64,21 +55,21 @@ const Caret = /*#__PURE__*/(0, _base.default)("div", {
|
|
|
64
55
|
} : {
|
|
65
56
|
name: "1lo1a34",
|
|
66
57
|
styles: "position:relative;top:2px",
|
|
67
|
-
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  return sortedData.map(node => {\n    const leaf = node.children.length <= 1 && !node.children[0]?.isDir && depth > 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 = depth === 0 || node.children.some(c => c.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"]} */",
|
|
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  return sortedData.map(node => {\n    const leaf = node.children.length === 0 && depth > 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 = depth === 0 || 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"]} */",
|
|
68
59
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
69
60
|
});
|
|
70
|
-
const CaretDown = /*#__PURE__*/(
|
|
61
|
+
const CaretDown = /*#__PURE__*/_styled(Caret, {
|
|
71
62
|
target: "eh84wlq2",
|
|
72
63
|
label: "CaretDown"
|
|
73
|
-
})(_decapCmsUiDefault.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  return sortedData.map(node => {\n    const leaf = node.children.length <= 1 && !node.children[0]?.isDir && depth > 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 = depth === 0 || node.children.some(c => c.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"]} */"));
|
|
74
|
-
const CaretRight = /*#__PURE__*/(
|
|
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  return sortedData.map(node => {\n    const leaf = node.children.length === 0 && depth > 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 = depth === 0 || 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"]} */"));
|
|
65
|
+
const CaretRight = /*#__PURE__*/_styled(Caret, {
|
|
75
66
|
target: "eh84wlq1",
|
|
76
67
|
label: "CaretRight"
|
|
77
|
-
})(_decapCmsUiDefault.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  return sortedData.map(node => {\n    const leaf = node.children.length <= 1 && !node.children[0]?.isDir && depth > 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 = depth === 0 || node.children.some(c => c.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"]} */"));
|
|
78
|
-
const TreeNavLink = /*#__PURE__*/(
|
|
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  return sortedData.map(node => {\n    const leaf = node.children.length === 0 && depth > 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 = depth === 0 || 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"]} */"));
|
|
69
|
+
const TreeNavLink = /*#__PURE__*/_styled(NavLink, {
|
|
79
70
|
target: "eh84wlq0",
|
|
80
71
|
label: "TreeNavLink"
|
|
81
|
-
})("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;", _decapCmsUiDefault.Icon, "{margin-right:4px;flex-shrink:0;}", props => /*#__PURE__*/(0, _react2.css)("&:hover,&:active,&.", props.activeClassName, "{color:", _decapCmsUiDefault.colors.active, ";background-color:", _decapCmsUiDefault.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  return sortedData.map(node => {\n    const leaf = node.children.length <= 1 && !node.children[0]?.isDir && depth > 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 = depth === 0 || node.children.some(c => c.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  return sortedData.map(node => {\n    const leaf = node.children.length <= 1 && !node.children[0]?.isDir && depth > 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 = depth === 0 || node.children.some(c => c.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"]} */"));
|
|
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  return sortedData.map(node => {\n    const leaf = node.children.length === 0 && depth > 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 = depth === 0 || 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  return sortedData.map(node => {\n    const leaf = node.children.length === 0 && depth > 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 = depth === 0 || 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"]} */"));
|
|
82
73
|
function getNodeTitle(node) {
|
|
83
74
|
var _node$children$find;
|
|
84
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;
|
|
@@ -92,10 +83,9 @@ function TreeNode(props) {
|
|
|
92
83
|
onToggle
|
|
93
84
|
} = props;
|
|
94
85
|
const collectionName = collection.get('name');
|
|
95
|
-
const sortedData = (
|
|
86
|
+
const sortedData = _sortBy(treeData, getNodeTitle);
|
|
96
87
|
return sortedData.map(node => {
|
|
97
|
-
|
|
98
|
-
const leaf = node.children.length <= 1 && !((_node$children$ = node.children[0]) !== null && _node$children$ !== void 0 && _node$children$.isDir) && depth > 0;
|
|
88
|
+
const leaf = node.children.length === 0 && depth > 0;
|
|
99
89
|
if (leaf) {
|
|
100
90
|
return null;
|
|
101
91
|
}
|
|
@@ -104,10 +94,10 @@ function TreeNode(props) {
|
|
|
104
94
|
to = `${to}/filter${node.path}`;
|
|
105
95
|
}
|
|
106
96
|
const title = getNodeTitle(node);
|
|
107
|
-
const hasChildren = depth === 0 || node.children.some(c => c.
|
|
108
|
-
return (
|
|
97
|
+
const hasChildren = depth === 0 || node.children.some(c => c.isDir);
|
|
98
|
+
return ___EmotionJSX(React.Fragment, {
|
|
109
99
|
key: node.path
|
|
110
|
-
}, (
|
|
100
|
+
}, ___EmotionJSX(TreeNavLink, {
|
|
111
101
|
exact: true,
|
|
112
102
|
to: to,
|
|
113
103
|
activeClassName: "sidebar-active",
|
|
@@ -117,9 +107,9 @@ function TreeNode(props) {
|
|
|
117
107
|
}),
|
|
118
108
|
depth: depth,
|
|
119
109
|
"data-testid": node.path
|
|
120
|
-
}, (
|
|
110
|
+
}, ___EmotionJSX(Icon, {
|
|
121
111
|
type: "write"
|
|
122
|
-
}), (
|
|
112
|
+
}), ___EmotionJSX(NodeTitleContainer, null, ___EmotionJSX(NodeTitle, null, title), hasChildren && (node.expanded ? ___EmotionJSX(CaretDown, null) : ___EmotionJSX(CaretRight, null)))), node.expanded && ___EmotionJSX(TreeNode, {
|
|
123
113
|
collection: collection,
|
|
124
114
|
depth: depth + 1,
|
|
125
115
|
treeData: node.children,
|
|
@@ -128,12 +118,12 @@ function TreeNode(props) {
|
|
|
128
118
|
});
|
|
129
119
|
}
|
|
130
120
|
TreeNode.propTypes = {
|
|
131
|
-
collection:
|
|
132
|
-
depth:
|
|
133
|
-
treeData:
|
|
134
|
-
onToggle:
|
|
121
|
+
collection: ImmutablePropTypes.map.isRequired,
|
|
122
|
+
depth: PropTypes.number,
|
|
123
|
+
treeData: PropTypes.array.isRequired,
|
|
124
|
+
onToggle: PropTypes.func.isRequired
|
|
135
125
|
};
|
|
136
|
-
function walk(treeData, callback) {
|
|
126
|
+
export function walk(treeData, callback) {
|
|
137
127
|
function traverse(children) {
|
|
138
128
|
for (const child of children) {
|
|
139
129
|
callback(child);
|
|
@@ -142,18 +132,18 @@ function walk(treeData, callback) {
|
|
|
142
132
|
}
|
|
143
133
|
return traverse(treeData);
|
|
144
134
|
}
|
|
145
|
-
function getTreeData(collection, entries) {
|
|
135
|
+
export function getTreeData(collection, entries) {
|
|
146
136
|
const collectionFolder = collection.get('folder');
|
|
147
137
|
const rootFolder = '/';
|
|
148
138
|
const entriesObj = entries.toJS().map(e => _objectSpread(_objectSpread({}, e), {}, {
|
|
149
139
|
path: e.path.slice(collectionFolder.length)
|
|
150
140
|
}));
|
|
151
141
|
const dirs = entriesObj.reduce((acc, entry) => {
|
|
152
|
-
let dir =
|
|
142
|
+
let dir = dirname(entry.path);
|
|
153
143
|
while (!acc[dir] && dir && dir !== rootFolder) {
|
|
154
|
-
const parts = dir.split(
|
|
144
|
+
const parts = dir.split(sep);
|
|
155
145
|
acc[dir] = parts.pop();
|
|
156
|
-
dir = parts.length && parts.join(
|
|
146
|
+
dir = parts.length && parts.join(sep);
|
|
157
147
|
}
|
|
158
148
|
return acc;
|
|
159
149
|
}, {});
|
|
@@ -175,7 +165,7 @@ function getTreeData(collection, entries) {
|
|
|
175
165
|
})), ...entriesObj.map((e, index) => {
|
|
176
166
|
let entryMap = entries.get(index);
|
|
177
167
|
entryMap = entryMap.set('data', addFileTemplateFields(entryMap.get('path'), entryMap.get('data')));
|
|
178
|
-
const title =
|
|
168
|
+
const title = selectEntryCollectionTitle(collection, entryMap);
|
|
179
169
|
return _objectSpread(_objectSpread({}, e), {}, {
|
|
180
170
|
title,
|
|
181
171
|
isDir: false,
|
|
@@ -183,7 +173,7 @@ function getTreeData(collection, entries) {
|
|
|
183
173
|
});
|
|
184
174
|
})];
|
|
185
175
|
const parentsToChildren = flatData.reduce((acc, node) => {
|
|
186
|
-
const parent = node.path === rootFolder ? '' :
|
|
176
|
+
const parent = node.path === rootFolder ? '' : dirname(node.path);
|
|
187
177
|
if (acc[parent]) {
|
|
188
178
|
acc[parent].push(node);
|
|
189
179
|
} else {
|
|
@@ -205,7 +195,7 @@ function getTreeData(collection, entries) {
|
|
|
205
195
|
const treeData = parentsToChildren[''].reduce(reducer, []);
|
|
206
196
|
return treeData;
|
|
207
197
|
}
|
|
208
|
-
function updateNode(treeData, node, callback) {
|
|
198
|
+
export function updateNode(treeData, node, callback) {
|
|
209
199
|
let stop = false;
|
|
210
200
|
function updater(nodes) {
|
|
211
201
|
if (stop) {
|
|
@@ -223,7 +213,7 @@ function updateNode(treeData, node, callback) {
|
|
|
223
213
|
}
|
|
224
214
|
return updater([...treeData]);
|
|
225
215
|
}
|
|
226
|
-
class NestedCollection extends
|
|
216
|
+
export class NestedCollection extends React.Component {
|
|
227
217
|
constructor(props) {
|
|
228
218
|
super(props);
|
|
229
219
|
_defineProperty(this, "onToggle", ({
|
|
@@ -285,26 +275,25 @@ class NestedCollection extends _react.default.Component {
|
|
|
285
275
|
const {
|
|
286
276
|
collection
|
|
287
277
|
} = this.props;
|
|
288
|
-
return (
|
|
278
|
+
return ___EmotionJSX(TreeNode, {
|
|
289
279
|
collection: collection,
|
|
290
280
|
treeData: treeData,
|
|
291
281
|
onToggle: this.onToggle
|
|
292
282
|
});
|
|
293
283
|
}
|
|
294
284
|
}
|
|
295
|
-
exports.NestedCollection = NestedCollection;
|
|
296
285
|
_defineProperty(NestedCollection, "propTypes", {
|
|
297
|
-
collection:
|
|
298
|
-
entries:
|
|
299
|
-
filterTerm:
|
|
286
|
+
collection: ImmutablePropTypes.map.isRequired,
|
|
287
|
+
entries: ImmutablePropTypes.list.isRequired,
|
|
288
|
+
filterTerm: PropTypes.string
|
|
300
289
|
});
|
|
301
290
|
function mapStateToProps(state, ownProps) {
|
|
302
291
|
const {
|
|
303
292
|
collection
|
|
304
293
|
} = ownProps;
|
|
305
|
-
const entries =
|
|
294
|
+
const entries = selectEntries(state.entries, collection) || List();
|
|
306
295
|
return {
|
|
307
296
|
entries
|
|
308
297
|
};
|
|
309
298
|
}
|
|
310
|
-
|
|
299
|
+
export default connect(mapStateToProps, null)(NestedCollection);
|