decap-cms-core 3.7.1 → 3.8.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/decap-cms-core.js +17 -17
- package/dist/decap-cms-core.js.map +1 -1
- package/dist/esm/bootstrap.js +2 -2
- package/dist/esm/components/App/App.js +6 -3
- package/dist/esm/components/App/Header.js +38 -14
- package/dist/esm/components/Collection/Entries/Entries.js +14 -5
- package/dist/esm/components/Collection/Entries/EntriesCollection.js +65 -9
- package/dist/esm/components/Collection/Entries/EntriesSearch.js +13 -5
- package/dist/esm/components/Collection/Entries/EntryCard.js +84 -28
- package/dist/esm/components/Collection/Entries/EntryListing.js +46 -11
- package/dist/esm/components/UI/ErrorBoundary.js +2 -2
- package/dist/esm/constants/configSchema.js +14 -0
- package/dist/esm/lib/i18n.js +16 -0
- package/index.d.ts +5 -1
- package/package.json +2 -2
- package/src/components/App/App.js +2 -0
- package/src/components/App/Header.js +27 -0
- package/src/components/Collection/Entries/Entries.js +9 -0
- package/src/components/Collection/Entries/EntriesCollection.js +100 -4
- package/src/components/Collection/Entries/EntriesSearch.js +11 -3
- package/src/components/Collection/Entries/EntryCard.js +82 -3
- package/src/components/Collection/Entries/EntryListing.js +65 -5
- package/src/components/Collection/Entries/__tests__/EntriesCollection.spec.js +26 -18
- package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +1 -0
- package/src/constants/configSchema.js +9 -1
- package/src/lib/i18n.ts +21 -0
- package/src/types/redux.ts +5 -1
|
@@ -8,18 +8,20 @@ import partial from 'lodash/partial';
|
|
|
8
8
|
import { Cursor } from 'decap-cms-lib-util';
|
|
9
9
|
import { colors } from 'decap-cms-ui-default';
|
|
10
10
|
import { loadEntries as actionLoadEntries, traverseCollectionCursor as actionTraverseCollectionCursor } from '../../../actions/entries';
|
|
11
|
+
import { loadUnpublishedEntries } from '../../../actions/editorialWorkflow';
|
|
11
12
|
import { selectEntries, selectEntriesLoaded, selectIsFetching, selectGroups } from '../../../reducers/entries';
|
|
13
|
+
import { selectUnpublishedEntry, selectUnpublishedEntriesByStatus } from '../../../reducers';
|
|
12
14
|
import { selectCollectionEntriesCursor } from '../../../reducers/cursors';
|
|
13
15
|
import Entries from './Entries';
|
|
14
16
|
import { jsx as ___EmotionJSX } from "@emotion/react";
|
|
15
17
|
const GroupHeading = /*#__PURE__*/_styled("h2", {
|
|
16
18
|
target: "eucqz2q1",
|
|
17
19
|
label: "GroupHeading"
|
|
18
|
-
})("font-size:22px;font-weight:600;line-height:37px;padding-inline-start:20px;color:", colors.textLead, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Collection/Entries/EntriesCollection.js"],"names":[],"mappings":"AAuB8B","file":"../../../../../src/components/Collection/Entries/EntriesCollection.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { connect } from 'react-redux';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport partial from 'lodash/partial';\nimport { Cursor } from 'decap-cms-lib-util';\nimport { colors } from 'decap-cms-ui-default';\n\nimport {\n  loadEntries as actionLoadEntries,\n  traverseCollectionCursor as actionTraverseCollectionCursor,\n} from '../../../actions/entries';\nimport {\n  selectEntries,\n  selectEntriesLoaded,\n  selectIsFetching,\n  selectGroups,\n} from '../../../reducers/entries';\nimport { selectCollectionEntriesCursor } from '../../../reducers/cursors';\nimport Entries from './Entries';\n\nconst GroupHeading = styled.h2`\n  font-size: 22px;\n  font-weight: 600;\n  line-height: 37px;\n  padding-inline-start: 20px;\n  color: ${colors.textLead};\n`;\n\nconst GroupContainer = styled.div``;\n\nfunction getGroupEntries(entries, paths) {\n  return entries.filter(entry => paths.has(entry.get('path')));\n}\n\nfunction getGroupTitle(group, t) {\n  const { label, value } = group;\n  if (value === undefined) {\n    return t('collection.groups.other');\n  }\n  if (typeof value === 'boolean') {\n    return value ? label : t('collection.groups.negateLabel', { label });\n  }\n  return `${label} ${value}`.trim();\n}\n\nfunction withGroups(groups, entries, EntriesToRender, t) {\n  return groups.map(group => {\n    const title = getGroupTitle(group, t);\n    return (\n      <GroupContainer key={group.id} id={group.id}>\n        <GroupHeading>{title}</GroupHeading>\n        <EntriesToRender entries={getGroupEntries(entries, group.paths)} />\n      </GroupContainer>\n    );\n  });\n}\n\nexport class EntriesCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    page: PropTypes.number,\n    entries: ImmutablePropTypes.list,\n    groups: PropTypes.array,\n    isFetching: PropTypes.bool.isRequired,\n    viewStyle: PropTypes.string,\n    cursor: PropTypes.object.isRequired,\n    loadEntries: PropTypes.func.isRequired,\n    traverseCollectionCursor: PropTypes.func.isRequired,\n    entriesLoaded: PropTypes.bool,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EntriesCollection.propTypes, this.props, 'prop', 'EntriesCollection');\n\n    const { collection, entriesLoaded, loadEntries } = this.props;\n    if (collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entriesLoaded, loadEntries } = this.props;\n    if (collection !== prevProps.collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n  }\n\n  handleCursorActions = (cursor, action) => {\n    const { collection, traverseCollectionCursor } = this.props;\n    traverseCollectionCursor(collection, action);\n  };\n\n  render() {\n    const { collection, entries, groups, isFetching, viewStyle, cursor, page, t } = this.props;\n\n    const EntriesToRender = ({ entries }) => {\n      return (\n        <Entries\n          collections={collection}\n          entries={entries}\n          isFetching={isFetching}\n          collectionName={collection.get('label')}\n          viewStyle={viewStyle}\n          cursor={cursor}\n          handleCursorActions={partial(this.handleCursorActions, cursor)}\n          page={page}\n        />\n      );\n    };\n\n    if (groups && groups.length > 0) {\n      return withGroups(groups, entries, EntriesToRender, t);\n    }\n\n    return <EntriesToRender entries={entries} />;\n  }\n}\n\nexport function filterNestedEntries(path, collectionFolder, entries, subfolders) {\n  const filtered = entries.filter(e => {\n    let entryPath = e.get('path').slice(collectionFolder.length + 1);\n    if (!entryPath.startsWith(path)) {\n      return false;\n    }\n\n    // for subdirectories, trim off the parent folder corresponding to\n    // this nested collection entry\n    if (path) {\n      entryPath = entryPath.slice(path.length + 1);\n    }\n\n    // if subfolders legacy mode is enabled, show only immediate subfolders\n    // also show index file in root folder\n    if (subfolders) {\n      const depth = entryPath.split('/').length;\n      return path ? depth === 2 : depth <= 2;\n    }\n\n    // only show immediate children\n    return !entryPath.includes('/');\n  });\n  return filtered;\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection, viewStyle, filterTerm } = ownProps;\n  const page = state.entries.getIn(['pages', collection.get('name'), 'page']);\n\n  let entries = selectEntries(state.entries, collection);\n  const groups = selectGroups(state.entries, collection);\n\n  if (collection.has('nested')) {\n    const collectionFolder = collection.get('folder');\n    entries = filterNestedEntries(\n      filterTerm || '',\n      collectionFolder,\n      entries,\n      collection.get('nested').get('subfolders') !== false,\n    );\n  }\n  const entriesLoaded = selectEntriesLoaded(state.entries, collection.get('name'));\n  const isFetching = selectIsFetching(state.entries, collection.get('name'));\n\n  const rawCursor = selectCollectionEntriesCursor(state.cursors, collection.get('name'));\n  const cursor = Cursor.create(rawCursor).clearData();\n\n  return { collection, page, entries, groups, entriesLoaded, isFetching, viewStyle, cursor };\n}\n\nconst mapDispatchToProps = {\n  loadEntries: actionLoadEntries,\n  traverseCollectionCursor: actionTraverseCollectionCursor,\n};\n\nconst ConnectedEntriesCollection = connect(mapStateToProps, mapDispatchToProps)(EntriesCollection);\n\nexport default translate()(ConnectedEntriesCollection);\n"]} */"));
|
|
20
|
+
})("font-size:22px;font-weight:600;line-height:37px;padding-inline-start:20px;color:", colors.textLead, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Collection/Entries/EntriesCollection.js"],"names":[],"mappings":"AAyB8B","file":"../../../../../src/components/Collection/Entries/EntriesCollection.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { connect } from 'react-redux';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport partial from 'lodash/partial';\nimport { Cursor } from 'decap-cms-lib-util';\nimport { colors } from 'decap-cms-ui-default';\n\nimport {\n  loadEntries as actionLoadEntries,\n  traverseCollectionCursor as actionTraverseCollectionCursor,\n} from '../../../actions/entries';\nimport { loadUnpublishedEntries } from '../../../actions/editorialWorkflow';\nimport {\n  selectEntries,\n  selectEntriesLoaded,\n  selectIsFetching,\n  selectGroups,\n} from '../../../reducers/entries';\nimport { selectUnpublishedEntry, selectUnpublishedEntriesByStatus } from '../../../reducers';\nimport { selectCollectionEntriesCursor } from '../../../reducers/cursors';\nimport Entries from './Entries';\n\nconst GroupHeading = styled.h2`\n  font-size: 22px;\n  font-weight: 600;\n  line-height: 37px;\n  padding-inline-start: 20px;\n  color: ${colors.textLead};\n`;\n\nconst GroupContainer = styled.div``;\n\nfunction getGroupEntries(entries, paths) {\n  return entries.filter(entry => paths.has(entry.get('path')));\n}\n\nfunction getGroupTitle(group, t) {\n  const { label, value } = group;\n  if (value === undefined) {\n    return t('collection.groups.other');\n  }\n  if (typeof value === 'boolean') {\n    return value ? label : t('collection.groups.negateLabel', { label });\n  }\n  return `${label} ${value}`.trim();\n}\n\nfunction withGroups(groups, entries, EntriesToRender, t) {\n  return groups.map(group => {\n    const title = getGroupTitle(group, t);\n    return (\n      <GroupContainer key={group.id} id={group.id}>\n        <GroupHeading>{title}</GroupHeading>\n        <EntriesToRender entries={getGroupEntries(entries, group.paths)} />\n      </GroupContainer>\n    );\n  });\n}\n\nexport class EntriesCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    collections: ImmutablePropTypes.iterable,\n    page: PropTypes.number,\n    entries: ImmutablePropTypes.list,\n    groups: PropTypes.array,\n    isFetching: PropTypes.bool.isRequired,\n    viewStyle: PropTypes.string,\n    cursor: PropTypes.object.isRequired,\n    loadEntries: PropTypes.func.isRequired,\n    traverseCollectionCursor: PropTypes.func.isRequired,\n    entriesLoaded: PropTypes.bool,\n    loadUnpublishedEntries: PropTypes.func.isRequired,\n    unpublishedEntriesLoaded: PropTypes.bool,\n    isEditorialWorkflowEnabled: PropTypes.bool,\n    getWorkflowStatus: PropTypes.func.isRequired,\n    getUnpublishedEntries: PropTypes.func.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EntriesCollection.propTypes, this.props, 'prop', 'EntriesCollection');\n\n    const {\n      collection,\n      collections,\n      entriesLoaded,\n      loadEntries,\n      unpublishedEntriesLoaded,\n      loadUnpublishedEntries,\n      isEditorialWorkflowEnabled,\n    } = this.props;\n\n    if (collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n\n    if (isEditorialWorkflowEnabled && !unpublishedEntriesLoaded) {\n      loadUnpublishedEntries(collections);\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const {\n      collection,\n      collections,\n      entriesLoaded,\n      loadEntries,\n      unpublishedEntriesLoaded,\n      loadUnpublishedEntries,\n      isEditorialWorkflowEnabled,\n    } = this.props;\n\n    if (collection !== prevProps.collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n\n    if (\n      isEditorialWorkflowEnabled &&\n      (!unpublishedEntriesLoaded || collection !== prevProps.collection)\n    ) {\n      loadUnpublishedEntries(collections);\n    }\n  }\n\n  handleCursorActions = (cursor, action) => {\n    const { collection, traverseCollectionCursor } = this.props;\n    traverseCollectionCursor(collection, action);\n  };\n\n  render() {\n    const {\n      collection,\n      entries,\n      groups,\n      isFetching,\n      viewStyle,\n      cursor,\n      page,\n      t,\n      getWorkflowStatus,\n      getUnpublishedEntries,\n      filterTerm,\n    } = this.props;\n\n    const EntriesToRender = ({ entries }) => {\n      return (\n        <Entries\n          collections={collection}\n          entries={entries}\n          isFetching={isFetching}\n          collectionName={collection.get('label')}\n          viewStyle={viewStyle}\n          cursor={cursor}\n          handleCursorActions={partial(this.handleCursorActions, cursor)}\n          page={page}\n          getWorkflowStatus={getWorkflowStatus}\n          getUnpublishedEntries={getUnpublishedEntries}\n          filterTerm={filterTerm}\n        />\n      );\n    };\n\n    if (groups && groups.length > 0) {\n      return withGroups(groups, entries, EntriesToRender, t);\n    }\n\n    return <EntriesToRender entries={entries} />;\n  }\n}\n\nexport function filterNestedEntries(path, collectionFolder, entries, subfolders) {\n  const filtered = entries.filter(e => {\n    let entryPath = e.get('path').slice(collectionFolder.length + 1);\n    if (!entryPath.startsWith(path)) {\n      return false;\n    }\n\n    // for subdirectories, trim off the parent folder corresponding to\n    // this nested collection entry\n    if (path) {\n      entryPath = entryPath.slice(path.length + 1);\n    }\n\n    // if subfolders legacy mode is enabled, show only immediate subfolders\n    // also show index file in root folder\n    if (subfolders) {\n      const depth = entryPath.split('/').length;\n      return path ? depth === 2 : depth <= 2;\n    }\n\n    // only show immediate children\n    return !entryPath.includes('/');\n  });\n  return filtered;\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection, viewStyle, filterTerm } = ownProps;\n  const page = state.entries.getIn(['pages', collection.get('name'), 'page']);\n\n  const collections = state.collections;\n\n  let entries = selectEntries(state.entries, collection);\n  const groups = selectGroups(state.entries, collection);\n\n  if (collection.has('nested')) {\n    const collectionFolder = collection.get('folder');\n    entries = filterNestedEntries(\n      filterTerm || '',\n      collectionFolder,\n      entries,\n      collection.get('nested').get('subfolders') !== false,\n    );\n  }\n  const entriesLoaded = selectEntriesLoaded(state.entries, collection.get('name'));\n  const isFetching = selectIsFetching(state.entries, collection.get('name'));\n\n  const rawCursor = selectCollectionEntriesCursor(state.cursors, collection.get('name'));\n  const cursor = Cursor.create(rawCursor).clearData();\n\n  const isEditorialWorkflowEnabled = state.config?.publish_mode === 'editorial_workflow';\n  const unpublishedEntriesLoaded = isEditorialWorkflowEnabled\n    ? !!state.editorialWorkflow?.getIn(['pages', 'ids'], false)\n    : true;\n\n  return {\n    collection,\n    collections,\n    page,\n    entries,\n    groups,\n    entriesLoaded,\n    isFetching,\n    viewStyle,\n    cursor,\n    unpublishedEntriesLoaded,\n    isEditorialWorkflowEnabled,\n    getWorkflowStatus: (collectionName, slug) => {\n      const unpublishedEntry = selectUnpublishedEntry(state, collectionName, slug);\n      return unpublishedEntry ? unpublishedEntry.get('status') : null;\n    },\n    getUnpublishedEntries: collectionName => {\n      if (!isEditorialWorkflowEnabled) return [];\n\n      const allStatuses = ['draft', 'pending_review', 'pending_publish'];\n      const unpublishedEntries = [];\n\n      allStatuses.forEach(statusKey => {\n        const entriesForStatus = selectUnpublishedEntriesByStatus(state, statusKey);\n        if (entriesForStatus) {\n          entriesForStatus.forEach(entry => {\n            if (entry.get('collection') === collectionName) {\n              const entryWithCollection = entry.set('collection', collectionName);\n              unpublishedEntries.push(entryWithCollection);\n            }\n          });\n        }\n      });\n\n      return unpublishedEntries;\n    },\n  };\n}\n\nconst mapDispatchToProps = {\n  loadEntries: actionLoadEntries,\n  traverseCollectionCursor: actionTraverseCollectionCursor,\n  loadUnpublishedEntries: collections => loadUnpublishedEntries(collections),\n};\n\nconst ConnectedEntriesCollection = connect(mapStateToProps, mapDispatchToProps)(EntriesCollection);\n\nexport default translate()(ConnectedEntriesCollection);\n"]} */"));
|
|
19
21
|
const GroupContainer = /*#__PURE__*/_styled("div", {
|
|
20
22
|
target: "eucqz2q0",
|
|
21
23
|
label: "GroupContainer"
|
|
22
|
-
})(process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Collection/Entries/EntriesCollection.js"],"names":[],"mappings":"AA+BiC","file":"../../../../../src/components/Collection/Entries/EntriesCollection.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { connect } from 'react-redux';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport partial from 'lodash/partial';\nimport { Cursor } from 'decap-cms-lib-util';\nimport { colors } from 'decap-cms-ui-default';\n\nimport {\n  loadEntries as actionLoadEntries,\n  traverseCollectionCursor as actionTraverseCollectionCursor,\n} from '../../../actions/entries';\nimport {\n  selectEntries,\n  selectEntriesLoaded,\n  selectIsFetching,\n  selectGroups,\n} from '../../../reducers/entries';\nimport { selectCollectionEntriesCursor } from '../../../reducers/cursors';\nimport Entries from './Entries';\n\nconst GroupHeading = styled.h2`\n  font-size: 22px;\n  font-weight: 600;\n  line-height: 37px;\n  padding-inline-start: 20px;\n  color: ${colors.textLead};\n`;\n\nconst GroupContainer = styled.div``;\n\nfunction getGroupEntries(entries, paths) {\n  return entries.filter(entry => paths.has(entry.get('path')));\n}\n\nfunction getGroupTitle(group, t) {\n  const { label, value } = group;\n  if (value === undefined) {\n    return t('collection.groups.other');\n  }\n  if (typeof value === 'boolean') {\n    return value ? label : t('collection.groups.negateLabel', { label });\n  }\n  return `${label} ${value}`.trim();\n}\n\nfunction withGroups(groups, entries, EntriesToRender, t) {\n  return groups.map(group => {\n    const title = getGroupTitle(group, t);\n    return (\n      <GroupContainer key={group.id} id={group.id}>\n        <GroupHeading>{title}</GroupHeading>\n        <EntriesToRender entries={getGroupEntries(entries, group.paths)} />\n      </GroupContainer>\n    );\n  });\n}\n\nexport class EntriesCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    page: PropTypes.number,\n    entries: ImmutablePropTypes.list,\n    groups: PropTypes.array,\n    isFetching: PropTypes.bool.isRequired,\n    viewStyle: PropTypes.string,\n    cursor: PropTypes.object.isRequired,\n    loadEntries: PropTypes.func.isRequired,\n    traverseCollectionCursor: PropTypes.func.isRequired,\n    entriesLoaded: PropTypes.bool,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EntriesCollection.propTypes, this.props, 'prop', 'EntriesCollection');\n\n    const { collection, entriesLoaded, loadEntries } = this.props;\n    if (collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { collection, entriesLoaded, loadEntries } = this.props;\n    if (collection !== prevProps.collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n  }\n\n  handleCursorActions = (cursor, action) => {\n    const { collection, traverseCollectionCursor } = this.props;\n    traverseCollectionCursor(collection, action);\n  };\n\n  render() {\n    const { collection, entries, groups, isFetching, viewStyle, cursor, page, t } = this.props;\n\n    const EntriesToRender = ({ entries }) => {\n      return (\n        <Entries\n          collections={collection}\n          entries={entries}\n          isFetching={isFetching}\n          collectionName={collection.get('label')}\n          viewStyle={viewStyle}\n          cursor={cursor}\n          handleCursorActions={partial(this.handleCursorActions, cursor)}\n          page={page}\n        />\n      );\n    };\n\n    if (groups && groups.length > 0) {\n      return withGroups(groups, entries, EntriesToRender, t);\n    }\n\n    return <EntriesToRender entries={entries} />;\n  }\n}\n\nexport function filterNestedEntries(path, collectionFolder, entries, subfolders) {\n  const filtered = entries.filter(e => {\n    let entryPath = e.get('path').slice(collectionFolder.length + 1);\n    if (!entryPath.startsWith(path)) {\n      return false;\n    }\n\n    // for subdirectories, trim off the parent folder corresponding to\n    // this nested collection entry\n    if (path) {\n      entryPath = entryPath.slice(path.length + 1);\n    }\n\n    // if subfolders legacy mode is enabled, show only immediate subfolders\n    // also show index file in root folder\n    if (subfolders) {\n      const depth = entryPath.split('/').length;\n      return path ? depth === 2 : depth <= 2;\n    }\n\n    // only show immediate children\n    return !entryPath.includes('/');\n  });\n  return filtered;\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection, viewStyle, filterTerm } = ownProps;\n  const page = state.entries.getIn(['pages', collection.get('name'), 'page']);\n\n  let entries = selectEntries(state.entries, collection);\n  const groups = selectGroups(state.entries, collection);\n\n  if (collection.has('nested')) {\n    const collectionFolder = collection.get('folder');\n    entries = filterNestedEntries(\n      filterTerm || '',\n      collectionFolder,\n      entries,\n      collection.get('nested').get('subfolders') !== false,\n    );\n  }\n  const entriesLoaded = selectEntriesLoaded(state.entries, collection.get('name'));\n  const isFetching = selectIsFetching(state.entries, collection.get('name'));\n\n  const rawCursor = selectCollectionEntriesCursor(state.cursors, collection.get('name'));\n  const cursor = Cursor.create(rawCursor).clearData();\n\n  return { collection, page, entries, groups, entriesLoaded, isFetching, viewStyle, cursor };\n}\n\nconst mapDispatchToProps = {\n  loadEntries: actionLoadEntries,\n  traverseCollectionCursor: actionTraverseCollectionCursor,\n};\n\nconst ConnectedEntriesCollection = connect(mapStateToProps, mapDispatchToProps)(EntriesCollection);\n\nexport default translate()(ConnectedEntriesCollection);\n"]} */");
|
|
24
|
+
})(process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Collection/Entries/EntriesCollection.js"],"names":[],"mappings":"AAiCiC","file":"../../../../../src/components/Collection/Entries/EntriesCollection.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { connect } from 'react-redux';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport partial from 'lodash/partial';\nimport { Cursor } from 'decap-cms-lib-util';\nimport { colors } from 'decap-cms-ui-default';\n\nimport {\n  loadEntries as actionLoadEntries,\n  traverseCollectionCursor as actionTraverseCollectionCursor,\n} from '../../../actions/entries';\nimport { loadUnpublishedEntries } from '../../../actions/editorialWorkflow';\nimport {\n  selectEntries,\n  selectEntriesLoaded,\n  selectIsFetching,\n  selectGroups,\n} from '../../../reducers/entries';\nimport { selectUnpublishedEntry, selectUnpublishedEntriesByStatus } from '../../../reducers';\nimport { selectCollectionEntriesCursor } from '../../../reducers/cursors';\nimport Entries from './Entries';\n\nconst GroupHeading = styled.h2`\n  font-size: 22px;\n  font-weight: 600;\n  line-height: 37px;\n  padding-inline-start: 20px;\n  color: ${colors.textLead};\n`;\n\nconst GroupContainer = styled.div``;\n\nfunction getGroupEntries(entries, paths) {\n  return entries.filter(entry => paths.has(entry.get('path')));\n}\n\nfunction getGroupTitle(group, t) {\n  const { label, value } = group;\n  if (value === undefined) {\n    return t('collection.groups.other');\n  }\n  if (typeof value === 'boolean') {\n    return value ? label : t('collection.groups.negateLabel', { label });\n  }\n  return `${label} ${value}`.trim();\n}\n\nfunction withGroups(groups, entries, EntriesToRender, t) {\n  return groups.map(group => {\n    const title = getGroupTitle(group, t);\n    return (\n      <GroupContainer key={group.id} id={group.id}>\n        <GroupHeading>{title}</GroupHeading>\n        <EntriesToRender entries={getGroupEntries(entries, group.paths)} />\n      </GroupContainer>\n    );\n  });\n}\n\nexport class EntriesCollection extends React.Component {\n  static propTypes = {\n    collection: ImmutablePropTypes.map.isRequired,\n    collections: ImmutablePropTypes.iterable,\n    page: PropTypes.number,\n    entries: ImmutablePropTypes.list,\n    groups: PropTypes.array,\n    isFetching: PropTypes.bool.isRequired,\n    viewStyle: PropTypes.string,\n    cursor: PropTypes.object.isRequired,\n    loadEntries: PropTypes.func.isRequired,\n    traverseCollectionCursor: PropTypes.func.isRequired,\n    entriesLoaded: PropTypes.bool,\n    loadUnpublishedEntries: PropTypes.func.isRequired,\n    unpublishedEntriesLoaded: PropTypes.bool,\n    isEditorialWorkflowEnabled: PropTypes.bool,\n    getWorkflowStatus: PropTypes.func.isRequired,\n    getUnpublishedEntries: PropTypes.func.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EntriesCollection.propTypes, this.props, 'prop', 'EntriesCollection');\n\n    const {\n      collection,\n      collections,\n      entriesLoaded,\n      loadEntries,\n      unpublishedEntriesLoaded,\n      loadUnpublishedEntries,\n      isEditorialWorkflowEnabled,\n    } = this.props;\n\n    if (collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n\n    if (isEditorialWorkflowEnabled && !unpublishedEntriesLoaded) {\n      loadUnpublishedEntries(collections);\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const {\n      collection,\n      collections,\n      entriesLoaded,\n      loadEntries,\n      unpublishedEntriesLoaded,\n      loadUnpublishedEntries,\n      isEditorialWorkflowEnabled,\n    } = this.props;\n\n    if (collection !== prevProps.collection && !entriesLoaded) {\n      loadEntries(collection);\n    }\n\n    if (\n      isEditorialWorkflowEnabled &&\n      (!unpublishedEntriesLoaded || collection !== prevProps.collection)\n    ) {\n      loadUnpublishedEntries(collections);\n    }\n  }\n\n  handleCursorActions = (cursor, action) => {\n    const { collection, traverseCollectionCursor } = this.props;\n    traverseCollectionCursor(collection, action);\n  };\n\n  render() {\n    const {\n      collection,\n      entries,\n      groups,\n      isFetching,\n      viewStyle,\n      cursor,\n      page,\n      t,\n      getWorkflowStatus,\n      getUnpublishedEntries,\n      filterTerm,\n    } = this.props;\n\n    const EntriesToRender = ({ entries }) => {\n      return (\n        <Entries\n          collections={collection}\n          entries={entries}\n          isFetching={isFetching}\n          collectionName={collection.get('label')}\n          viewStyle={viewStyle}\n          cursor={cursor}\n          handleCursorActions={partial(this.handleCursorActions, cursor)}\n          page={page}\n          getWorkflowStatus={getWorkflowStatus}\n          getUnpublishedEntries={getUnpublishedEntries}\n          filterTerm={filterTerm}\n        />\n      );\n    };\n\n    if (groups && groups.length > 0) {\n      return withGroups(groups, entries, EntriesToRender, t);\n    }\n\n    return <EntriesToRender entries={entries} />;\n  }\n}\n\nexport function filterNestedEntries(path, collectionFolder, entries, subfolders) {\n  const filtered = entries.filter(e => {\n    let entryPath = e.get('path').slice(collectionFolder.length + 1);\n    if (!entryPath.startsWith(path)) {\n      return false;\n    }\n\n    // for subdirectories, trim off the parent folder corresponding to\n    // this nested collection entry\n    if (path) {\n      entryPath = entryPath.slice(path.length + 1);\n    }\n\n    // if subfolders legacy mode is enabled, show only immediate subfolders\n    // also show index file in root folder\n    if (subfolders) {\n      const depth = entryPath.split('/').length;\n      return path ? depth === 2 : depth <= 2;\n    }\n\n    // only show immediate children\n    return !entryPath.includes('/');\n  });\n  return filtered;\n}\n\nfunction mapStateToProps(state, ownProps) {\n  const { collection, viewStyle, filterTerm } = ownProps;\n  const page = state.entries.getIn(['pages', collection.get('name'), 'page']);\n\n  const collections = state.collections;\n\n  let entries = selectEntries(state.entries, collection);\n  const groups = selectGroups(state.entries, collection);\n\n  if (collection.has('nested')) {\n    const collectionFolder = collection.get('folder');\n    entries = filterNestedEntries(\n      filterTerm || '',\n      collectionFolder,\n      entries,\n      collection.get('nested').get('subfolders') !== false,\n    );\n  }\n  const entriesLoaded = selectEntriesLoaded(state.entries, collection.get('name'));\n  const isFetching = selectIsFetching(state.entries, collection.get('name'));\n\n  const rawCursor = selectCollectionEntriesCursor(state.cursors, collection.get('name'));\n  const cursor = Cursor.create(rawCursor).clearData();\n\n  const isEditorialWorkflowEnabled = state.config?.publish_mode === 'editorial_workflow';\n  const unpublishedEntriesLoaded = isEditorialWorkflowEnabled\n    ? !!state.editorialWorkflow?.getIn(['pages', 'ids'], false)\n    : true;\n\n  return {\n    collection,\n    collections,\n    page,\n    entries,\n    groups,\n    entriesLoaded,\n    isFetching,\n    viewStyle,\n    cursor,\n    unpublishedEntriesLoaded,\n    isEditorialWorkflowEnabled,\n    getWorkflowStatus: (collectionName, slug) => {\n      const unpublishedEntry = selectUnpublishedEntry(state, collectionName, slug);\n      return unpublishedEntry ? unpublishedEntry.get('status') : null;\n    },\n    getUnpublishedEntries: collectionName => {\n      if (!isEditorialWorkflowEnabled) return [];\n\n      const allStatuses = ['draft', 'pending_review', 'pending_publish'];\n      const unpublishedEntries = [];\n\n      allStatuses.forEach(statusKey => {\n        const entriesForStatus = selectUnpublishedEntriesByStatus(state, statusKey);\n        if (entriesForStatus) {\n          entriesForStatus.forEach(entry => {\n            if (entry.get('collection') === collectionName) {\n              const entryWithCollection = entry.set('collection', collectionName);\n              unpublishedEntries.push(entryWithCollection);\n            }\n          });\n        }\n      });\n\n      return unpublishedEntries;\n    },\n  };\n}\n\nconst mapDispatchToProps = {\n  loadEntries: actionLoadEntries,\n  traverseCollectionCursor: actionTraverseCollectionCursor,\n  loadUnpublishedEntries: collections => loadUnpublishedEntries(collections),\n};\n\nconst ConnectedEntriesCollection = connect(mapStateToProps, mapDispatchToProps)(EntriesCollection);\n\nexport default translate()(ConnectedEntriesCollection);\n"]} */");
|
|
23
25
|
function getGroupEntries(entries, paths) {
|
|
24
26
|
return entries.filter(entry => paths.has(entry.get('path')));
|
|
25
27
|
}
|
|
@@ -52,6 +54,7 @@ function withGroups(groups, entries, EntriesToRender, t) {
|
|
|
52
54
|
export class EntriesCollection extends React.Component {
|
|
53
55
|
static propTypes = {
|
|
54
56
|
collection: ImmutablePropTypes.map.isRequired,
|
|
57
|
+
collections: ImmutablePropTypes.iterable,
|
|
55
58
|
page: PropTypes.number,
|
|
56
59
|
entries: ImmutablePropTypes.list,
|
|
57
60
|
groups: PropTypes.array,
|
|
@@ -60,29 +63,48 @@ export class EntriesCollection extends React.Component {
|
|
|
60
63
|
cursor: PropTypes.object.isRequired,
|
|
61
64
|
loadEntries: PropTypes.func.isRequired,
|
|
62
65
|
traverseCollectionCursor: PropTypes.func.isRequired,
|
|
63
|
-
entriesLoaded: PropTypes.bool
|
|
66
|
+
entriesLoaded: PropTypes.bool,
|
|
67
|
+
loadUnpublishedEntries: PropTypes.func.isRequired,
|
|
68
|
+
unpublishedEntriesLoaded: PropTypes.bool,
|
|
69
|
+
isEditorialWorkflowEnabled: PropTypes.bool,
|
|
70
|
+
getWorkflowStatus: PropTypes.func.isRequired,
|
|
71
|
+
getUnpublishedEntries: PropTypes.func.isRequired
|
|
64
72
|
};
|
|
65
73
|
componentDidMount() {
|
|
66
74
|
// Manually validate PropTypes - React 19 breaking change
|
|
67
75
|
PropTypes.checkPropTypes(EntriesCollection.propTypes, this.props, 'prop', 'EntriesCollection');
|
|
68
76
|
const {
|
|
69
77
|
collection,
|
|
78
|
+
collections,
|
|
70
79
|
entriesLoaded,
|
|
71
|
-
loadEntries
|
|
80
|
+
loadEntries,
|
|
81
|
+
unpublishedEntriesLoaded,
|
|
82
|
+
loadUnpublishedEntries,
|
|
83
|
+
isEditorialWorkflowEnabled
|
|
72
84
|
} = this.props;
|
|
73
85
|
if (collection && !entriesLoaded) {
|
|
74
86
|
loadEntries(collection);
|
|
75
87
|
}
|
|
88
|
+
if (isEditorialWorkflowEnabled && !unpublishedEntriesLoaded) {
|
|
89
|
+
loadUnpublishedEntries(collections);
|
|
90
|
+
}
|
|
76
91
|
}
|
|
77
92
|
componentDidUpdate(prevProps) {
|
|
78
93
|
const {
|
|
79
94
|
collection,
|
|
95
|
+
collections,
|
|
80
96
|
entriesLoaded,
|
|
81
|
-
loadEntries
|
|
97
|
+
loadEntries,
|
|
98
|
+
unpublishedEntriesLoaded,
|
|
99
|
+
loadUnpublishedEntries,
|
|
100
|
+
isEditorialWorkflowEnabled
|
|
82
101
|
} = this.props;
|
|
83
102
|
if (collection !== prevProps.collection && !entriesLoaded) {
|
|
84
103
|
loadEntries(collection);
|
|
85
104
|
}
|
|
105
|
+
if (isEditorialWorkflowEnabled && (!unpublishedEntriesLoaded || collection !== prevProps.collection)) {
|
|
106
|
+
loadUnpublishedEntries(collections);
|
|
107
|
+
}
|
|
86
108
|
}
|
|
87
109
|
handleCursorActions = (cursor, action) => {
|
|
88
110
|
const {
|
|
@@ -100,7 +122,10 @@ export class EntriesCollection extends React.Component {
|
|
|
100
122
|
viewStyle,
|
|
101
123
|
cursor,
|
|
102
124
|
page,
|
|
103
|
-
t
|
|
125
|
+
t,
|
|
126
|
+
getWorkflowStatus,
|
|
127
|
+
getUnpublishedEntries,
|
|
128
|
+
filterTerm
|
|
104
129
|
} = this.props;
|
|
105
130
|
const EntriesToRender = ({
|
|
106
131
|
entries
|
|
@@ -113,7 +138,10 @@ export class EntriesCollection extends React.Component {
|
|
|
113
138
|
viewStyle: viewStyle,
|
|
114
139
|
cursor: cursor,
|
|
115
140
|
handleCursorActions: partial(this.handleCursorActions, cursor),
|
|
116
|
-
page: page
|
|
141
|
+
page: page,
|
|
142
|
+
getWorkflowStatus: getWorkflowStatus,
|
|
143
|
+
getUnpublishedEntries: getUnpublishedEntries,
|
|
144
|
+
filterTerm: filterTerm
|
|
117
145
|
});
|
|
118
146
|
};
|
|
119
147
|
if (groups && groups.length > 0) {
|
|
@@ -156,6 +184,7 @@ function mapStateToProps(state, ownProps) {
|
|
|
156
184
|
filterTerm
|
|
157
185
|
} = ownProps;
|
|
158
186
|
const page = state.entries.getIn(['pages', collection.get('name'), 'page']);
|
|
187
|
+
const collections = state.collections;
|
|
159
188
|
let entries = selectEntries(state.entries, collection);
|
|
160
189
|
const groups = selectGroups(state.entries, collection);
|
|
161
190
|
if (collection.has('nested')) {
|
|
@@ -166,20 +195,47 @@ function mapStateToProps(state, ownProps) {
|
|
|
166
195
|
const isFetching = selectIsFetching(state.entries, collection.get('name'));
|
|
167
196
|
const rawCursor = selectCollectionEntriesCursor(state.cursors, collection.get('name'));
|
|
168
197
|
const cursor = Cursor.create(rawCursor).clearData();
|
|
198
|
+
const isEditorialWorkflowEnabled = state.config?.publish_mode === 'editorial_workflow';
|
|
199
|
+
const unpublishedEntriesLoaded = isEditorialWorkflowEnabled ? !!state.editorialWorkflow?.getIn(['pages', 'ids'], false) : true;
|
|
169
200
|
return {
|
|
170
201
|
collection,
|
|
202
|
+
collections,
|
|
171
203
|
page,
|
|
172
204
|
entries,
|
|
173
205
|
groups,
|
|
174
206
|
entriesLoaded,
|
|
175
207
|
isFetching,
|
|
176
208
|
viewStyle,
|
|
177
|
-
cursor
|
|
209
|
+
cursor,
|
|
210
|
+
unpublishedEntriesLoaded,
|
|
211
|
+
isEditorialWorkflowEnabled,
|
|
212
|
+
getWorkflowStatus: (collectionName, slug) => {
|
|
213
|
+
const unpublishedEntry = selectUnpublishedEntry(state, collectionName, slug);
|
|
214
|
+
return unpublishedEntry ? unpublishedEntry.get('status') : null;
|
|
215
|
+
},
|
|
216
|
+
getUnpublishedEntries: collectionName => {
|
|
217
|
+
if (!isEditorialWorkflowEnabled) return [];
|
|
218
|
+
const allStatuses = ['draft', 'pending_review', 'pending_publish'];
|
|
219
|
+
const unpublishedEntries = [];
|
|
220
|
+
allStatuses.forEach(statusKey => {
|
|
221
|
+
const entriesForStatus = selectUnpublishedEntriesByStatus(state, statusKey);
|
|
222
|
+
if (entriesForStatus) {
|
|
223
|
+
entriesForStatus.forEach(entry => {
|
|
224
|
+
if (entry.get('collection') === collectionName) {
|
|
225
|
+
const entryWithCollection = entry.set('collection', collectionName);
|
|
226
|
+
unpublishedEntries.push(entryWithCollection);
|
|
227
|
+
}
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
});
|
|
231
|
+
return unpublishedEntries;
|
|
232
|
+
}
|
|
178
233
|
};
|
|
179
234
|
}
|
|
180
235
|
const mapDispatchToProps = {
|
|
181
236
|
loadEntries: actionLoadEntries,
|
|
182
|
-
traverseCollectionCursor: actionTraverseCollectionCursor
|
|
237
|
+
traverseCollectionCursor: actionTraverseCollectionCursor,
|
|
238
|
+
loadUnpublishedEntries: collections => loadUnpublishedEntries(collections)
|
|
183
239
|
};
|
|
184
240
|
const ConnectedEntriesCollection = connect(mapStateToProps, mapDispatchToProps)(EntriesCollection);
|
|
185
241
|
export default translate()(ConnectedEntriesCollection);
|
|
@@ -4,7 +4,7 @@ import ImmutablePropTypes from 'react-immutable-proptypes';
|
|
|
4
4
|
import { connect } from 'react-redux';
|
|
5
5
|
import isEqual from 'lodash/isEqual';
|
|
6
6
|
import { Cursor } from 'decap-cms-lib-util';
|
|
7
|
-
import { selectSearchedEntries } from '../../../reducers';
|
|
7
|
+
import { selectSearchedEntries, selectUnpublishedEntry } from '../../../reducers';
|
|
8
8
|
import { searchEntries as actionSearchEntries, clearSearch as actionClearSearch } from '../../../actions/search';
|
|
9
9
|
import Entries from './Entries';
|
|
10
10
|
import { jsx as ___EmotionJSX } from "@emotion/react";
|
|
@@ -17,7 +17,8 @@ class EntriesSearch extends React.Component {
|
|
|
17
17
|
collections: ImmutablePropTypes.seq,
|
|
18
18
|
collectionNames: PropTypes.array,
|
|
19
19
|
entries: ImmutablePropTypes.list,
|
|
20
|
-
page: PropTypes.number
|
|
20
|
+
page: PropTypes.number,
|
|
21
|
+
getWorkflowStatus: PropTypes.func
|
|
21
22
|
};
|
|
22
23
|
componentDidMount() {
|
|
23
24
|
// Manually validate PropTypes - React 19 breaking change
|
|
@@ -69,14 +70,16 @@ class EntriesSearch extends React.Component {
|
|
|
69
70
|
const {
|
|
70
71
|
collections,
|
|
71
72
|
entries,
|
|
72
|
-
isFetching
|
|
73
|
+
isFetching,
|
|
74
|
+
getWorkflowStatus
|
|
73
75
|
} = this.props;
|
|
74
76
|
return ___EmotionJSX(Entries, {
|
|
75
77
|
cursor: this.getCursor(),
|
|
76
78
|
handleCursorActions: this.handleCursorActions,
|
|
77
79
|
collections: collections,
|
|
78
80
|
entries: entries,
|
|
79
|
-
isFetching: isFetching
|
|
81
|
+
isFetching: isFetching,
|
|
82
|
+
getWorkflowStatus: getWorkflowStatus
|
|
80
83
|
});
|
|
81
84
|
}
|
|
82
85
|
}
|
|
@@ -89,13 +92,18 @@ function mapStateToProps(state, ownProps) {
|
|
|
89
92
|
const isFetching = state.search.isFetching;
|
|
90
93
|
const page = state.search.page;
|
|
91
94
|
const entries = selectSearchedEntries(state, collectionNames);
|
|
95
|
+
function getWorkflowStatus(collectionName, slug) {
|
|
96
|
+
const unpublishedEntry = selectUnpublishedEntry(state, collectionName, slug);
|
|
97
|
+
return unpublishedEntry ? unpublishedEntry.get('status') : null;
|
|
98
|
+
}
|
|
92
99
|
return {
|
|
93
100
|
isFetching,
|
|
94
101
|
page,
|
|
95
102
|
collections,
|
|
96
103
|
collectionNames,
|
|
97
104
|
entries,
|
|
98
|
-
searchTerm
|
|
105
|
+
searchTerm,
|
|
106
|
+
getWorkflowStatus
|
|
99
107
|
};
|
|
100
108
|
}
|
|
101
109
|
const mapDispatchToProps = {
|