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.
@@ -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 = {